From b58cf15740180dfe27db3a9a97c7899c7060be8a Mon Sep 17 00:00:00 2001 From: Andrew Date: Wed, 19 Feb 2025 21:25:14 +0000 Subject: [PATCH 1/3] Added light sensor and on off switch --- .../examples/Zigbee_Basic_On_Off/README.md | 57 ++++ .../Zigbee_Basic_On_Off.ino | 103 +++++++ .../examples/Zigbee_Basic_On_Off/ci.json | 6 + .../examples/Zigbee_Light_Sensor/README.md | 79 ++++++ .../Zigbee_Light_Sensor.ino | 118 ++++++++ .../examples/Zigbee_Light_Sensor/ci.json | 6 + .../Zigbee_Temperature_Sensor.ino | 21 -- .../Zigbee_Thermostat/Zigbee_Thermostat.ino | 15 - libraries/Zigbee/library.properties | 2 +- libraries/Zigbee/src/Zigbee.h | 6 +- libraries/Zigbee/src/ZigbeeCore.cpp | 1 + libraries/Zigbee/src/ZigbeeCore.h | 2 +- libraries/Zigbee/src/ZigbeeEP.cpp | 195 +------------ libraries/Zigbee/src/ZigbeeEP.h | 75 ++--- libraries/Zigbee/src/ZigbeeHandlers.cpp | 263 +----------------- .../Zigbee/src/ep/ZigbeeBasicOnOffSensor.cpp | 52 ++++ .../Zigbee/src/ep/ZigbeeBasicOnOffSensor.h | 48 ++++ .../src/ep/ZigbeeCarbonDioxideSensor.cpp | 7 +- .../Zigbee/src/ep/ZigbeeCarbonDioxideSensor.h | 2 +- .../src/ep/ZigbeeColorDimmableLight.cpp | 116 ++++---- .../Zigbee/src/ep/ZigbeeColorDimmableLight.h | 62 +---- .../Zigbee/src/ep/ZigbeeColorDimmerSwitch.cpp | 50 +++- .../Zigbee/src/ep/ZigbeeColorDimmerSwitch.h | 2 +- libraries/Zigbee/src/ep/ZigbeeDimmableLight.h | 2 +- libraries/Zigbee/src/ep/ZigbeeFlowSensor.cpp | 1 + libraries/Zigbee/src/ep/ZigbeeFlowSensor.h | 2 +- libraries/Zigbee/src/ep/ZigbeeLight.h | 2 +- libraries/Zigbee/src/ep/ZigbeeLightSensor.cpp | 76 +++++ libraries/Zigbee/src/ep/ZigbeeLightSensor.h | 53 ++++ .../Zigbee/src/ep/ZigbeeOccupancySensor.cpp | 1 + .../Zigbee/src/ep/ZigbeeOccupancySensor.h | 2 +- .../Zigbee/src/ep/ZigbeePressureSensor.cpp | 3 +- .../Zigbee/src/ep/ZigbeePressureSensor.h | 2 +- libraries/Zigbee/src/ep/ZigbeeSwitch.h | 2 +- libraries/Zigbee/src/ep/ZigbeeTempSensor.h | 2 +- libraries/Zigbee/src/ep/ZigbeeThermostat.h | 2 +- 36 files changed, 751 insertions(+), 687 deletions(-) create mode 100644 libraries/Zigbee/examples/Zigbee_Basic_On_Off/README.md create mode 100644 libraries/Zigbee/examples/Zigbee_Basic_On_Off/Zigbee_Basic_On_Off.ino create mode 100644 libraries/Zigbee/examples/Zigbee_Basic_On_Off/ci.json create mode 100644 libraries/Zigbee/examples/Zigbee_Light_Sensor/README.md create mode 100644 libraries/Zigbee/examples/Zigbee_Light_Sensor/Zigbee_Light_Sensor.ino create mode 100644 libraries/Zigbee/examples/Zigbee_Light_Sensor/ci.json create mode 100644 libraries/Zigbee/src/ep/ZigbeeBasicOnOffSensor.cpp create mode 100644 libraries/Zigbee/src/ep/ZigbeeBasicOnOffSensor.h create mode 100644 libraries/Zigbee/src/ep/ZigbeeLightSensor.cpp create mode 100644 libraries/Zigbee/src/ep/ZigbeeLightSensor.h diff --git a/libraries/Zigbee/examples/Zigbee_Basic_On_Off/README.md b/libraries/Zigbee/examples/Zigbee_Basic_On_Off/README.md new file mode 100644 index 00000000000..0c5dcd013f2 --- /dev/null +++ b/libraries/Zigbee/examples/Zigbee_Basic_On_Off/README.md @@ -0,0 +1,57 @@ +# Arduino-ESP32 Zigbee Occupancy Sensor Example + +This example shows how to configure the Zigbee end device and use it as a Home Automation (HA) occupancy sensor (PIR). + +# Supported Targets + +Currently, this example supports the following targets. + +| Supported Targets | ESP32-C6 | ESP32-H2 | +| ----------------- | -------- | -------- | + +## Hardware Required + +* A USB cable for power supply and programming + +### Configure the Project + +Set the Button GPIO by changing the `button` variable. By default, it's the pin `BOOT_PIN` (BOOT button on ESP32-C6 and ESP32-H2). +Set the Sensor GPIO by changing the `sensor_pin` variable. + +#### Using Arduino IDE + +To get more information about the Espressif boards see [Espressif Development Kits](https://www.espressif.com/en/products/devkits). + +* Before Compile/Verify, select the correct board: `Tools -> Board`. +* Select the End device Zigbee mode: `Tools -> Zigbee mode: Zigbee ED (end device)` +* Select Partition Scheme for Zigbee: `Tools -> Partition Scheme: Zigbee 4MB with spiffs` +* Select the COM port: `Tools -> Port: xxx` where the `xxx` is the detected COM port. +* Optional: Set debug level to verbose to see all logs from Zigbee stack: `Tools -> Core Debug Level: Verbose`. + +## Troubleshooting + +If the End device flashed with this example is not connecting to the coordinator, erase the flash of the End device before flashing the example to the board. + +***Important: Make sure you are using a good quality USB cable and that you have a reliable power source*** + +* **LED not blinking:** Check the wiring connection and the IO selection. +* **Programming Fail:** If the programming/flash procedure fails, try reducing the serial connection speed. +* **COM port not detected:** Check the USB cable and the USB to Serial driver installation. + +If the error persists, you can ask for help at the official [ESP32 forum](https://esp32.com) or see [Contribute](#contribute). + +## Contribute + +To know how to contribute to this project, see [How to contribute.](https://github.com/espressif/arduino-esp32/blob/master/CONTRIBUTING.rst) + +If you have any **feedback** or **issue** to report on this example/library, please open an issue or fix it by creating a new PR. Contributions are more than welcome! + +Before creating a new issue, be sure to try Troubleshooting and check if the same issue was already created by someone else. + +## Resources + +* Official ESP32 Forum: [Link](https://esp32.com) +* Arduino-ESP32 Official Repository: [espressif/arduino-esp32](https://github.com/espressif/arduino-esp32) +* ESP32-C6 Datasheet: [Link to datasheet](https://www.espressif.com/sites/default/files/documentation/esp32-c6_datasheet_en.pdf) +* ESP32-H2 Datasheet: [Link to datasheet](https://www.espressif.com/sites/default/files/documentation/esp32-h2_datasheet_en.pdf) +* Official ESP-IDF documentation: [ESP-IDF](https://idf.espressif.com) diff --git a/libraries/Zigbee/examples/Zigbee_Basic_On_Off/Zigbee_Basic_On_Off.ino b/libraries/Zigbee/examples/Zigbee_Basic_On_Off/Zigbee_Basic_On_Off.ino new file mode 100644 index 00000000000..405e9518c10 --- /dev/null +++ b/libraries/Zigbee/examples/Zigbee_Basic_On_Off/Zigbee_Basic_On_Off.ino @@ -0,0 +1,103 @@ +// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +/** + * @brief This example demonstrates Zigbee occupancy sensor. + * + * The example demonstrates how to use Zigbee library to create a end device occupancy sensor. + * The occupancy sensor is a Zigbee end device, which is reporting data to the Zigbee network. + * Tested with PIR sensor HC-SR501 connected to GPIO4. + * + * Proper Zigbee mode must be selected in Tools->Zigbee mode + * and also the correct partition scheme must be selected in Tools->Partition Scheme. + * + * Please check the README.md for instructions and more detailed description. + * + * Created by Jan Procházka (https://github.com/P-R-O-C-H-Y/) + */ + +#ifndef ZIGBEE_MODE_ED +#error "Zigbee end device mode is not selected in Tools->Zigbee mode" +#endif + +#include "Zigbee.h" + +/* Zigbee on off sensor configuration */ +#define ON_OFF_SENSOR_ENDPOINT_NUMBER 10 +uint8_t button = BOOT_PIN; +uint8_t sensor_pin = 4; + +ZigbeeBasicOnOffSensor zbOnOffSensor = ZigbeeBasicOnOffSensor(ON_OFF_SENSOR_ENDPOINT_NUMBER); + +void setup() { + Serial.begin(115200); + + // Init button + PIR sensor + pinMode(button, INPUT_PULLUP); + pinMode(sensor_pin, INPUT); + + // Optional: set Zigbee device name and model + zbOnOffSensor.setManufacturerAndModel("Espressif", "ZigbeeOnOffSensor"); + + // Add endpoint to Zigbee Core + Zigbee.addEndpoint(&zbOnOffSensor); + + Serial.println("Starting Zigbee..."); + // When all EPs are registered, start Zigbee in End Device mode + if (!Zigbee.begin()) { + Serial.println("Zigbee failed to start!"); + Serial.println("Rebooting..."); + ESP.restart(); + } else { + Serial.println("Zigbee started successfully!"); + } + Serial.println("Connecting to network"); + while (!Zigbee.connected()) { + Serial.print("."); + delay(100); + } + Serial.println(); +} + +void loop() { + // Check input state + static bool state = false; + if (digitalRead(sensor_pin) == HIGH && !state) { + // Update state value + zbOnOffSensor.setState(true); + zbOnOffSensor.report(); + state = true; + } else if (digitalRead(sensor_pin) == LOW && state) { + zbOnOffSensor.setState(false); + zbOnOffSensor.report(); + state = false; + } + + // Checking button for factory reset + if (digitalRead(button) == LOW) { // Push button pressed + // Key debounce handling + delay(100); + int startTime = millis(); + while (digitalRead(button) == LOW) { + delay(50); + if ((millis() - startTime) > 3000) { + // If key pressed for more than 3secs, factory reset Zigbee and reboot + Serial.println("Resetting Zigbee to factory and rebooting in 1s."); + delay(1000); + Zigbee.factoryReset(); + } + } + } + delay(100); +} diff --git a/libraries/Zigbee/examples/Zigbee_Basic_On_Off/ci.json b/libraries/Zigbee/examples/Zigbee_Basic_On_Off/ci.json new file mode 100644 index 00000000000..7b7ccef8ed7 --- /dev/null +++ b/libraries/Zigbee/examples/Zigbee_Basic_On_Off/ci.json @@ -0,0 +1,6 @@ +{ + "fqbn_append": "PartitionScheme=zigbee,ZigbeeMode=ed", + "requires": [ + "CONFIG_SOC_IEEE802154_SUPPORTED=y" + ] +} diff --git a/libraries/Zigbee/examples/Zigbee_Light_Sensor/README.md b/libraries/Zigbee/examples/Zigbee_Light_Sensor/README.md new file mode 100644 index 00000000000..577bd7c8058 --- /dev/null +++ b/libraries/Zigbee/examples/Zigbee_Light_Sensor/README.md @@ -0,0 +1,79 @@ +# Arduino-ESP32 Zigbee Temperature Sensor Example + +This example shows how to configure the Zigbee end device and use it as a Home Automation (HA) temperature sensor. + +# Supported Targets + +Currently, this example supports the following targets. + +| Supported Targets | ESP32-C6 | ESP32-H2 | +| ----------------- | -------- | -------- | + +## Temperature Sensor Functions + +Note: + * This board means the board (e.g. ESP32-H2 / C6) loaded with `Zigbee_Temperature_Sensor` example. + * The remote board means the board (e.g. ESP32-H2 / C6) loaded with `Zigbee_Thermostat` example. + +Functions: + * After this board first starts up, it would be configured locally to report the temperature on 1 degree change and no periodic reporting to the remote board. + * By clicking the button (BOOT) on this board, this board will immediately send a report of the current measured temperature to the remote board. + +## Hardware Required + +* One development board (ESP32-H2 or ESP32-C6) acting as Zigbee coordinator (loaded with `Zigbee_Thermostat` example) +* A USB cable for power supply and programming +* Choose another board (ESP32-H2 or ESP32-C6) as Zigbee end device and upload the `Zigbee_Temperature_Sensor` example + +### Configure the Project + +In this example, the internal temperature sensor task is reading the chip temperature. +Set the Button GPIO by changing the `button` variable. By default, it's the pin `BOOT_PIN` (BOOT button on ESP32-C6 and ESP32-H2). + +#### Using Arduino IDE + +To get more information about the Espressif boards see [Espressif Development Kits](https://www.espressif.com/en/products/devkits). + +* Before Compile/Verify, select the correct board: `Tools -> Board`. +* Select the End device Zigbee mode: `Tools -> Zigbee mode: Zigbee ED (end device)` +* Select Partition Scheme for Zigbee: `Tools -> Partition Scheme: Zigbee 4MB with spiffs` +* Select the COM port: `Tools -> Port: xxx` where the `xxx` is the detected COM port. +* Optional: Set debug level to verbose to see all logs from Zigbee stack: `Tools -> Core Debug Level: Verbose`. + +## Troubleshooting + +If the End device flashed with this example is not connecting to the coordinator, erase the flash of the End device before flashing the example to the board. It is recommended to do this if you re-flash the coordinator. +You can do the following: + +* In the Arduino IDE go to the Tools menu and set `Erase All Flash Before Sketch Upload` to `Enabled`. +* Add to the sketch `Zigbee.factoryReset();` to reset the device and Zigbee stack. + +By default, the coordinator network is closed after rebooting or flashing new firmware. +To open the network you have 2 options: + +* Open network after reboot by setting `Zigbee.setRebootOpenNetwork(time);` before calling `Zigbee.begin();`. +* In application you can anytime call `Zigbee.openNetwork(time);` to open the network for devices to join. + +***Important: Make sure you are using a good quality USB cable and that you have a reliable power source*** + +* **LED not blinking:** Check the wiring connection and the IO selection. +* **Programming Fail:** If the programming/flash procedure fails, try reducing the serial connection speed. +* **COM port not detected:** Check the USB cable and the USB to Serial driver installation. + +If the error persists, you can ask for help at the official [ESP32 forum](https://esp32.com) or see [Contribute](#contribute). + +## Contribute + +To know how to contribute to this project, see [How to contribute.](https://github.com/espressif/arduino-esp32/blob/master/CONTRIBUTING.rst) + +If you have any **feedback** or **issue** to report on this example/library, please open an issue or fix it by creating a new PR. Contributions are more than welcome! + +Before creating a new issue, be sure to try Troubleshooting and check if the same issue was already created by someone else. + +## Resources + +* Official ESP32 Forum: [Link](https://esp32.com) +* Arduino-ESP32 Official Repository: [espressif/arduino-esp32](https://github.com/espressif/arduino-esp32) +* ESP32-C6 Datasheet: [Link to datasheet](https://www.espressif.com/sites/default/files/documentation/esp32-c6_datasheet_en.pdf) +* ESP32-H2 Datasheet: [Link to datasheet](https://www.espressif.com/sites/default/files/documentation/esp32-h2_datasheet_en.pdf) +* Official ESP-IDF documentation: [ESP-IDF](https://idf.espressif.com) diff --git a/libraries/Zigbee/examples/Zigbee_Light_Sensor/Zigbee_Light_Sensor.ino b/libraries/Zigbee/examples/Zigbee_Light_Sensor/Zigbee_Light_Sensor.ino new file mode 100644 index 00000000000..b82570d0c8d --- /dev/null +++ b/libraries/Zigbee/examples/Zigbee_Light_Sensor/Zigbee_Light_Sensor.ino @@ -0,0 +1,118 @@ +// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +/** + * @brief This example demonstrates a Zigbee light sensor. + * + * The example demonstrates how to use Zigbee library to create a end device light sensor. + * The light sensor is a Zigbee end device, which is controlled by a Zigbee coordinator. + * + * Proper Zigbee mode must be selected in Tools->Zigbee mode + * and also the correct partition scheme must be selected in Tools->Partition Scheme. + * + * Please check the README.md for instructions and more detailed description. + * + * Created by Andrew McDonald (https://youtube.com/techteamgb) + * Modified from examples by Jan Procházka (https://github.com/P-R-O-C-H-Y/) + */ + +#ifndef ZIGBEE_MODE_ED +#error "Zigbee end device mode is not selected in Tools->Zigbee mode" +#endif + +#include "Zigbee.h" + +/* Zigbee light sensor configuration */ +#define LIGHT_SENSOR_ENDPOINT_NUMBER 10 +uint8_t button = BOOT_PIN; + +ZigbeeLightSensor zbLightSensor = ZigbeeLightSensor(LIGHT_SENSOR_ENDPOINT_NUMBER); + +/************************ Light sensor *****************************/ +static void light_sensor_value_update(void *arg) { + for (;;) { + // Read light sensor value + uint16_t lsens_value = (uint16_t)temperatureRead(); + Serial.printf("Updated light sensor value to %d\r\n", lsens_value); + // Update illuminance value in light sensor EP + zbLightSensor.setIlluminance(lsens_value); + delay(1000); + } +} + +/********************* Arduino functions **************************/ +void setup() { + Serial.begin(115200); + + // Init button switch + pinMode(button, INPUT_PULLUP); + + // Optional: set Zigbee device name and model + zbLightSensor.setManufacturerAndModel("Espressif", "ZigbeeLightSensor"); + + // Set minimum and maximum light measurement value + zbLightSensor.setMinMaxValue(0, 100); + + // Optional: Set tolerance for light measurement + zbLightSensor.setTolerance(1); + + // Add endpoint to Zigbee Core + Zigbee.addEndpoint(&zbLightSensor); + + Serial.println("Starting Zigbee..."); + // When all EPs are registered, start Zigbee in End Device mode + if (!Zigbee.begin()) { + Serial.println("Zigbee failed to start!"); + Serial.println("Rebooting..."); + ESP.restart(); + } else { + Serial.println("Zigbee started successfully!"); + } + Serial.println("Connecting to network"); + while (!Zigbee.connected()) { + Serial.print("."); + delay(100); + } + Serial.println(); + + // Start light sensor reading task + xTaskCreate(light_sensor_value_update, "light_sensor_update", 2048, NULL, 10, NULL); + + // Set reporting interval for light measurement in seconds, must be called after Zigbee.begin() + // min_interval and max_interval in seconds, delta + // if min = 1 and max = 0, reporting is sent only when illuminance changes by delta + // if min = 0 and max = 10, reporting is sent every 10 seconds or illuminance changes by delta + // if min = 0, max = 10 and delta = 0, reporting is sent every 10 seconds regardless of illuminance change + zbLightSensor.setReporting(0, 30, 0); +} + +void loop() { + // Checking button for factory reset + if (digitalRead(button) == LOW) { // Push button pressed + // Key debounce handling + delay(100); + int startTime = millis(); + while (digitalRead(button) == LOW) { + delay(50); + if ((millis() - startTime) > 3000) { + // If key pressed for more than 3secs, factory reset Zigbee and reboot + Serial.println("Resetting Zigbee to factory and rebooting in 1s."); + delay(1000); + Zigbee.factoryReset(); + } + } + zbLightSensor.report(); + } + delay(100); +} diff --git a/libraries/Zigbee/examples/Zigbee_Light_Sensor/ci.json b/libraries/Zigbee/examples/Zigbee_Light_Sensor/ci.json new file mode 100644 index 00000000000..7b7ccef8ed7 --- /dev/null +++ b/libraries/Zigbee/examples/Zigbee_Light_Sensor/ci.json @@ -0,0 +1,6 @@ +{ + "fqbn_append": "PartitionScheme=zigbee,ZigbeeMode=ed", + "requires": [ + "CONFIG_SOC_IEEE802154_SUPPORTED=y" + ] +} diff --git a/libraries/Zigbee/examples/Zigbee_Temperature_Sensor/Zigbee_Temperature_Sensor.ino b/libraries/Zigbee/examples/Zigbee_Temperature_Sensor/Zigbee_Temperature_Sensor.ino index ad007abbbaa..27aa2db97bd 100644 --- a/libraries/Zigbee/examples/Zigbee_Temperature_Sensor/Zigbee_Temperature_Sensor.ino +++ b/libraries/Zigbee/examples/Zigbee_Temperature_Sensor/Zigbee_Temperature_Sensor.ino @@ -36,11 +36,6 @@ #define TEMP_SENSOR_ENDPOINT_NUMBER 10 uint8_t button = BOOT_PIN; -// Optional Time cluster variables -struct tm timeinfo; -struct tm *localTime; -int32_t timezone; - ZigbeeTempSensor zbTempSensor = ZigbeeTempSensor(TEMP_SENSOR_ENDPOINT_NUMBER); /************************ Temp sensor *****************************/ @@ -71,9 +66,6 @@ void setup() { // Optional: Set tolerance for temperature measurement in °C (lowest possible value is 0.01°C) zbTempSensor.setTolerance(1); - // Optional: Time cluster configuration (default params, as this device will revieve time from coordinator) - zbTempSensor.addTimeCluster(); - // Add endpoint to Zigbee Core Zigbee.addEndpoint(&zbTempSensor); @@ -93,19 +85,6 @@ void setup() { } Serial.println(); - // Optional: If time cluster is added, time can be read from the coordinator - timeinfo = zbTempSensor.getTime(); - timezone = zbTempSensor.getTimezone(); - - Serial.println("UTC time:"); - Serial.println(&timeinfo, "%A, %B %d %Y %H:%M:%S"); - - time_t local = mktime(&timeinfo) + timezone; - localTime = localtime(&local); - - Serial.println("Local time with timezone:"); - Serial.println(localTime, "%A, %B %d %Y %H:%M:%S"); - // Start Temperature sensor reading task xTaskCreate(temp_sensor_value_update, "temp_sensor_update", 2048, NULL, 10, NULL); diff --git a/libraries/Zigbee/examples/Zigbee_Thermostat/Zigbee_Thermostat.ino b/libraries/Zigbee/examples/Zigbee_Thermostat/Zigbee_Thermostat.ino index 7cdf45ef711..a4720feeba4 100644 --- a/libraries/Zigbee/examples/Zigbee_Thermostat/Zigbee_Thermostat.ino +++ b/libraries/Zigbee/examples/Zigbee_Thermostat/Zigbee_Thermostat.ino @@ -45,8 +45,6 @@ float sensor_max_temp; float sensor_min_temp; float sensor_tolerance; -struct tm timeinfo = {}; // Time structure for Time cluster - /****************** Temperature sensor handling *******************/ void recieveSensorTemp(float temperature) { Serial.printf("Temperature sensor value: %.2f°C\n", temperature); @@ -73,19 +71,6 @@ void setup() { //Optional: set Zigbee device name and model zbThermostat.setManufacturerAndModel("Espressif", "ZigbeeThermostat"); - //Optional Time cluster configuration - //example time January 13, 2025 13:30:30 CET - timeinfo.tm_year = 2025 - 1900; // = 2025 - timeinfo.tm_mon = 0; // January - timeinfo.tm_mday = 13; // 13th - timeinfo.tm_hour = 12; // 12 hours - 1 hour (CET) - timeinfo.tm_min = 30; // 30 minutes - timeinfo.tm_sec = 30; // 30 seconds - timeinfo.tm_isdst = -1; - - // Set time and gmt offset (timezone in seconds -> CET = +3600 seconds) - zbThermostat.addTimeCluster(timeinfo, 3600); - //Add endpoint to Zigbee Core Zigbee.addEndpoint(&zbThermostat); diff --git a/libraries/Zigbee/library.properties b/libraries/Zigbee/library.properties index 18fb81548c0..87e0cd63e9f 100644 --- a/libraries/Zigbee/library.properties +++ b/libraries/Zigbee/library.properties @@ -1,5 +1,5 @@ name=Zigbee -version=3.1.2 +version=3.1.1 author=P-R-O-C-H-Y maintainer=Jan Procházka sentence=Enables zigbee connection with the ESP32 diff --git a/libraries/Zigbee/src/Zigbee.h b/libraries/Zigbee/src/Zigbee.h index b785c28d0df..700d0f2243a 100644 --- a/libraries/Zigbee/src/Zigbee.h +++ b/libraries/Zigbee/src/Zigbee.h @@ -18,7 +18,5 @@ #include "ep/ZigbeeFlowSensor.h" #include "ep/ZigbeeOccupancySensor.h" #include "ep/ZigbeeCarbonDioxideSensor.h" -#include "ep/ZigbeeContactSwitch.h" -#include "ep/ZigbeeDoorWindowHandle.h" -#include "ep/ZigbeeWindowCovering.h" -#include "ep/ZigbeeVibrationSensor.h" +#include "ep/ZigbeeLightSensor.h" +#include "ep/ZigbeeBasicOnOffSensor.h" diff --git a/libraries/Zigbee/src/ZigbeeCore.cpp b/libraries/Zigbee/src/ZigbeeCore.cpp index 19f4d0872b8..0a3177919da 100644 --- a/libraries/Zigbee/src/ZigbeeCore.cpp +++ b/libraries/Zigbee/src/ZigbeeCore.cpp @@ -29,6 +29,7 @@ ZigbeeCore::ZigbeeCore() { } } } +ZigbeeCore::~ZigbeeCore() {} //forward declaration static esp_err_t zb_action_handler(esp_zb_core_action_callback_id_t callback_id, const void *message); diff --git a/libraries/Zigbee/src/ZigbeeCore.h b/libraries/Zigbee/src/ZigbeeCore.h index 02dce54e5ff..c72a60d81c4 100644 --- a/libraries/Zigbee/src/ZigbeeCore.h +++ b/libraries/Zigbee/src/ZigbeeCore.h @@ -86,7 +86,7 @@ class ZigbeeCore { public: ZigbeeCore(); - ~ZigbeeCore() {} + ~ZigbeeCore(); std::list ep_objects; diff --git a/libraries/Zigbee/src/ZigbeeEP.cpp b/libraries/Zigbee/src/ZigbeeEP.cpp index f3f890393eb..61f2fa8d2de 100644 --- a/libraries/Zigbee/src/ZigbeeEP.cpp +++ b/libraries/Zigbee/src/ZigbeeEP.cpp @@ -27,6 +27,8 @@ ZigbeeEP::ZigbeeEP(uint8_t endpoint) { } } +ZigbeeEP::~ZigbeeEP() {} + void ZigbeeEP::setVersion(uint8_t version) { _ep_config.app_device_version = version; } @@ -238,197 +240,4 @@ void ZigbeeEP::zbIdentify(const esp_zb_zcl_set_attr_value_message_t *message) { } } -void ZigbeeEP::addTimeCluster(tm time, int32_t gmt_offset) { - time_t utc_time = 0; - - // Check if time is set - if (time.tm_year > 0) { - // Convert time to UTC - utc_time = mktime(&time); - } - - // Create time cluster server attributes - esp_zb_attribute_list_t *time_cluster_server = esp_zb_zcl_attr_list_create(ESP_ZB_ZCL_CLUSTER_ID_TIME); - esp_zb_time_cluster_add_attr(time_cluster_server, ESP_ZB_ZCL_ATTR_TIME_TIME_ZONE_ID, (void *)&gmt_offset); - esp_zb_time_cluster_add_attr(time_cluster_server, ESP_ZB_ZCL_ATTR_TIME_TIME_ID, (void *)&utc_time); - // Create time cluster client attributes - esp_zb_attribute_list_t *time_cluster_client = esp_zb_zcl_attr_list_create(ESP_ZB_ZCL_CLUSTER_ID_TIME); - // Add time clusters to cluster list - esp_zb_cluster_list_add_time_cluster(_cluster_list, time_cluster_server, ESP_ZB_ZCL_CLUSTER_SERVER_ROLE); - esp_zb_cluster_list_add_time_cluster(_cluster_list, time_cluster_client, ESP_ZB_ZCL_CLUSTER_CLIENT_ROLE); -} - -void ZigbeeEP::setTime(tm time) { - time_t utc_time = mktime(&time); - esp_zb_lock_acquire(portMAX_DELAY); - esp_zb_zcl_set_attribute_val(_endpoint, ESP_ZB_ZCL_CLUSTER_ID_TIME, ESP_ZB_ZCL_CLUSTER_SERVER_ROLE, ESP_ZB_ZCL_ATTR_TIME_TIME_ID, &utc_time, false); - esp_zb_lock_release(); -} - -void ZigbeeEP::setTimezone(int32_t gmt_offset) { - esp_zb_lock_acquire(portMAX_DELAY); - esp_zb_zcl_set_attribute_val(_endpoint, ESP_ZB_ZCL_CLUSTER_ID_TIME, ESP_ZB_ZCL_CLUSTER_SERVER_ROLE, ESP_ZB_ZCL_ATTR_TIME_TIME_ZONE_ID, &gmt_offset, false); - esp_zb_lock_release(); -} - -tm ZigbeeEP::getTime(uint8_t endpoint, int32_t short_addr, esp_zb_ieee_addr_t ieee_addr) { - /* Read peer time */ - esp_zb_zcl_read_attr_cmd_t read_req; - - if (short_addr >= 0) { - read_req.address_mode = ESP_ZB_APS_ADDR_MODE_16_ENDP_PRESENT; - read_req.zcl_basic_cmd.dst_addr_u.addr_short = (uint16_t)short_addr; - } else { - read_req.address_mode = ESP_ZB_APS_ADDR_MODE_64_ENDP_PRESENT; - memcpy(read_req.zcl_basic_cmd.dst_addr_u.addr_long, ieee_addr, sizeof(esp_zb_ieee_addr_t)); - } - - uint16_t attributes[] = {ESP_ZB_ZCL_ATTR_TIME_TIME_ID}; - read_req.attr_number = ZB_ARRAY_LENTH(attributes); - read_req.attr_field = attributes; - - read_req.clusterID = ESP_ZB_ZCL_CLUSTER_ID_TIME; - - read_req.zcl_basic_cmd.dst_endpoint = endpoint; - read_req.zcl_basic_cmd.src_endpoint = _endpoint; - - // clear read time - _read_time = 0; - - log_v("Reading time from endpoint %d", endpoint); - esp_zb_zcl_read_attr_cmd_req(&read_req); - - //Wait for response or timeout - if (xSemaphoreTake(lock, ZB_CMD_TIMEOUT) != pdTRUE) { - log_e("Error while reading time"); - return tm(); - } - - struct tm *timeinfo = localtime(&_read_time); - if (timeinfo) { - return *timeinfo; - } else { - log_e("Error while converting time"); - return tm(); - } -} - -int32_t ZigbeeEP::getTimezone(uint8_t endpoint, int32_t short_addr, esp_zb_ieee_addr_t ieee_addr) { - /* Read peer timezone */ - esp_zb_zcl_read_attr_cmd_t read_req; - - if (short_addr >= 0) { - read_req.address_mode = ESP_ZB_APS_ADDR_MODE_16_ENDP_PRESENT; - read_req.zcl_basic_cmd.dst_addr_u.addr_short = (uint16_t)short_addr; - } else { - read_req.address_mode = ESP_ZB_APS_ADDR_MODE_64_ENDP_PRESENT; - memcpy(read_req.zcl_basic_cmd.dst_addr_u.addr_long, ieee_addr, sizeof(esp_zb_ieee_addr_t)); - } - - uint16_t attributes[] = {ESP_ZB_ZCL_ATTR_TIME_TIME_ZONE_ID}; - read_req.attr_number = ZB_ARRAY_LENTH(attributes); - read_req.attr_field = attributes; - - read_req.clusterID = ESP_ZB_ZCL_CLUSTER_ID_TIME; - - read_req.zcl_basic_cmd.dst_endpoint = endpoint; - read_req.zcl_basic_cmd.src_endpoint = _endpoint; - - // clear read timezone - _read_timezone = 0; - - log_v("Reading timezone from endpoint %d", endpoint); - esp_zb_zcl_read_attr_cmd_req(&read_req); - - //Wait for response or timeout - if (xSemaphoreTake(lock, ZB_CMD_TIMEOUT) != pdTRUE) { - log_e("Error while reading timezone"); - } - - return _read_timezone; -} - -void ZigbeeEP::zbReadTimeCluster(const esp_zb_zcl_attribute_t *attribute) { - /* Time cluster attributes */ - if (attribute->id == ESP_ZB_ZCL_ATTR_TIME_TIME_ID && attribute->data.type == ESP_ZB_ZCL_ATTR_TYPE_UTC_TIME) { - log_v("Time attribute received"); - log_v("Time: %lld", *(uint32_t *)attribute->data.value); - _read_time = *(uint32_t *)attribute->data.value; - xSemaphoreGive(lock); - } else if (attribute->id == ESP_ZB_ZCL_ATTR_TIME_TIME_ZONE_ID && attribute->data.type == ESP_ZB_ZCL_ATTR_TYPE_S32) { - log_v("Timezone attribute received"); - log_v("Timezone: %d", *(int32_t *)attribute->data.value); - _read_timezone = *(int32_t *)attribute->data.value; - xSemaphoreGive(lock); - } -} - -// typedef struct esp_zb_ota_cluster_cfg_s { -// uint32_t ota_upgrade_file_version; /*!< The attribute indicates the file version of the running firmware image on the device */ -// uint16_t ota_upgrade_manufacturer; /*!< The attribute indicates the value for the manufacturer of the device */ -// uint16_t ota_upgrade_image_type; /*!< The attribute indicates the the image type of the file that the client is currently downloading */ -// uint32_t ota_upgrade_downloaded_file_ver; /*!< The attribute indicates the file version of the downloaded image on the device*/ -// esp_zb_ota_cluster_cfg_t; - -// typedef struct esp_zb_zcl_ota_upgrade_client_variable_s { -// uint16_t timer_query; /*!< The field indicates the time of querying OTA image for OTA upgrade client */ -// uint16_t hw_version; /*!< The hardware version */ -// uint8_t max_data_size; /*!< The maximum size of OTA data */ -// } esp_zb_zcl_ota_upgrade_client_variable_t; - -void ZigbeeEP::addOTAClient( - uint32_t file_version, uint32_t downloaded_file_ver, uint16_t hw_version, uint16_t manufacturer, uint16_t image_type, uint8_t max_data_size -) { - - esp_zb_ota_cluster_cfg_t ota_cluster_cfg = {}; - ota_cluster_cfg.ota_upgrade_file_version = file_version; //OTA_UPGRADE_RUNNING_FILE_VERSION; - ota_cluster_cfg.ota_upgrade_downloaded_file_ver = downloaded_file_ver; //OTA_UPGRADE_DOWNLOADED_FILE_VERSION; - ota_cluster_cfg.ota_upgrade_manufacturer = manufacturer; //OTA_UPGRADE_MANUFACTURER; - ota_cluster_cfg.ota_upgrade_image_type = image_type; //OTA_UPGRADE_IMAGE_TYPE; - - esp_zb_attribute_list_t *ota_cluster = esp_zb_ota_cluster_create(&ota_cluster_cfg); - - esp_zb_zcl_ota_upgrade_client_variable_t variable_config = {}; - variable_config.timer_query = ESP_ZB_ZCL_OTA_UPGRADE_QUERY_TIMER_COUNT_DEF; - variable_config.hw_version = hw_version; //OTA_UPGRADE_HW_VERSION; - variable_config.max_data_size = max_data_size; //OTA_UPGRADE_MAX_DATA_SIZE; - - uint16_t ota_upgrade_server_addr = 0xffff; - uint8_t ota_upgrade_server_ep = 0xff; - - ESP_ERROR_CHECK(esp_zb_ota_cluster_add_attr(ota_cluster, ESP_ZB_ZCL_ATTR_OTA_UPGRADE_CLIENT_DATA_ID, (void *)&variable_config)); - ESP_ERROR_CHECK(esp_zb_ota_cluster_add_attr(ota_cluster, ESP_ZB_ZCL_ATTR_OTA_UPGRADE_SERVER_ADDR_ID, (void *)&ota_upgrade_server_addr)); - ESP_ERROR_CHECK(esp_zb_ota_cluster_add_attr(ota_cluster, ESP_ZB_ZCL_ATTR_OTA_UPGRADE_SERVER_ENDPOINT_ID, (void *)&ota_upgrade_server_ep)); - - ESP_ERROR_CHECK(esp_zb_cluster_list_add_ota_cluster(_cluster_list, ota_cluster, ESP_ZB_ZCL_CLUSTER_CLIENT_ROLE)); -} - -static void findOTAServer(esp_zb_zdp_status_t zdo_status, uint16_t addr, uint8_t endpoint, void *user_ctx) { - if (zdo_status == ESP_ZB_ZDP_STATUS_SUCCESS) { - esp_zb_ota_upgrade_client_query_interval_set(*((uint8_t *)user_ctx), OTA_UPGRADE_QUERY_INTERVAL); - esp_zb_ota_upgrade_client_query_image_req(addr, endpoint); - log_i("Query OTA upgrade from server endpoint: %d after %d seconds", endpoint, OTA_UPGRADE_QUERY_INTERVAL); - } else { - log_w("No OTA Server found"); - } -} - -void ZigbeeEP::requestOTAUpdate() { - esp_zb_zdo_match_desc_req_param_t req; - uint16_t cluster_list[] = {ESP_ZB_ZCL_CLUSTER_ID_OTA_UPGRADE}; - - /* Match the OTA server of coordinator */ - req.addr_of_interest = 0x0000; - req.dst_nwk_addr = 0x0000; - req.num_in_clusters = 1; - req.num_out_clusters = 0; - req.profile_id = ESP_ZB_AF_HA_PROFILE_ID; - req.cluster_list = cluster_list; - esp_zb_lock_acquire(portMAX_DELAY); - if (esp_zb_bdb_dev_joined()) { - esp_zb_zdo_match_cluster(&req, findOTAServer, &_endpoint); - } - esp_zb_lock_release(); -} - #endif //SOC_IEEE802154_SUPPORTED && CONFIG_ZB_ENABLED diff --git a/libraries/Zigbee/src/ZigbeeEP.h b/libraries/Zigbee/src/ZigbeeEP.h index 3bdd7f22b23..a5e9efa4283 100644 --- a/libraries/Zigbee/src/ZigbeeEP.h +++ b/libraries/Zigbee/src/ZigbeeEP.h @@ -6,13 +6,26 @@ #if SOC_IEEE802154_SUPPORTED && CONFIG_ZB_ENABLED #include -#include /* Useful defines */ -#define ZB_CMD_TIMEOUT 10000 // 10 seconds -#define OTA_UPGRADE_QUERY_INTERVAL (1 * 60) // 1 hour = 60 minutes +#define ZB_CMD_TIMEOUT 10000 // 10 seconds #define ZB_ARRAY_LENTH(arr) (sizeof(arr) / sizeof(arr[0])) +#define XYZ_TO_RGB(X, Y, Z, r, g, b) \ + { \ + r = (float)(3.240479 * (X) - 1.537150 * (Y) - 0.498535 * (Z)); \ + g = (float)(-0.969256 * (X) + 1.875992 * (Y) + 0.041556 * (Z)); \ + b = (float)(0.055648 * (X) - 0.204043 * (Y) + 1.057311 * (Z)); \ + if (r > 1) { \ + r = 1; \ + } \ + if (g > 1) { \ + g = 1; \ + } \ + if (b > 1) { \ + b = 1; \ + } \ + } #define RGB_TO_XYZ(r, g, b, X, Y, Z) \ { \ @@ -42,7 +55,7 @@ typedef enum { class ZigbeeEP { public: ZigbeeEP(uint8_t endpoint = 10); - ~ZigbeeEP() {} + ~ZigbeeEP(); // Set ep config and cluster list void setEpConfig(esp_zb_endpoint_config_t ep_config, esp_zb_cluster_list_t *cluster_list) { @@ -69,51 +82,20 @@ class ZigbeeEP { _allow_multiple_binding = bind; } - // Set Manufacturer name and model + // Manufacturer name and model implemented void setManufacturerAndModel(const char *name, const char *model); - - // Methods to read manufacturer and model name from selected endpoint and short address - char *readManufacturer(uint8_t endpoint, uint16_t short_addr, esp_zb_ieee_addr_t ieee_addr); - char *readModel(uint8_t endpoint, uint16_t short_addr, esp_zb_ieee_addr_t ieee_addr); - - // Set Power source and battery percentage for battery powered devices void setPowerSource(zb_power_source_t power_source, uint8_t percentage = 255); void setBatteryPercentage(uint8_t percentage); void reportBatteryPercentage(); - // Set time - void addTimeCluster(tm time = {}, int32_t gmt_offset = 0); // gmt offset in seconds - void setTime(tm time); - void setTimezone(int32_t gmt_offset); - - // Get time from Coordinator or specific endpoint (blocking until response) - struct tm getTime(uint8_t endpoint = 1, int32_t short_addr = 0x0000, esp_zb_ieee_addr_t ieee_addr = {0}); - int32_t getTimezone(uint8_t endpoint = 1, int32_t short_addr = 0x0000, esp_zb_ieee_addr_t ieee_addr = {0}); // gmt offset in seconds + // Methods to read manufacturer and model name from selected endpoint and short address + char *readManufacturer(uint8_t endpoint, uint16_t short_addr, esp_zb_ieee_addr_t ieee_addr); + char *readModel(uint8_t endpoint, uint16_t short_addr, esp_zb_ieee_addr_t ieee_addr); bool epAllowMultipleBinding() { return _allow_multiple_binding; } - // OTA methods - /** - * @brief Add OTA client to the Zigbee endpoint. - * - * @param file_version The current file version of the OTA client. - * @param downloaded_file_ver The version of the downloaded file. - * @param hw_version The hardware version of the device. - * @param manufacturer The manufacturer code (default: 0x1001). - * @param image_type The image type code (default: 0x1011). - * @param max_data_size The maximum data size for OTA transfer (default and recommended: 223). - */ - void addOTAClient( - uint32_t file_version, uint32_t downloaded_file_ver, uint16_t hw_version, uint16_t manufacturer = 0x1001, uint16_t image_type = 0x1011, - uint8_t max_data_size = 223 - ); - /** - * @brief Request OTA update from the server, first request is within a minute and the next requests are sent every hour automatically. - */ - void requestOTAUpdate(); - // findEndpoind may be implemented by EPs to find and bind devices virtual void findEndpoint(esp_zb_zdo_match_desc_req_param_t *cmd_req) {}; @@ -122,15 +104,6 @@ class ZigbeeEP { virtual void zbAttributeRead(uint16_t cluster_id, const esp_zb_zcl_attribute_t *attribute) {}; virtual void zbReadBasicCluster(const esp_zb_zcl_attribute_t *attribute); //already implemented virtual void zbIdentify(const esp_zb_zcl_set_attr_value_message_t *message); - virtual void zbWindowCoveringMovementCmd(const esp_zb_zcl_window_covering_movement_message_t *message) {}; - virtual void zbReadTimeCluster(const esp_zb_zcl_attribute_t *attribute); //already implemented - virtual void zbIASZoneStatusChangeNotification(const esp_zb_zcl_ias_zone_status_change_notification_message_t *message) {}; - virtual void zbIASZoneEnrollResponse(const esp_zb_zcl_ias_zone_enroll_response_message_t *message) {}; - - virtual void addBoundDevice(zb_device_params_t *device) { - _bound_devices.push_back(device); - _is_bound = true; - } void onIdentify(void (*callback)(uint16_t)) { _on_identify = callback; @@ -140,8 +113,6 @@ class ZigbeeEP { char *_read_manufacturer; char *_read_model; void (*_on_identify)(uint16_t time); - time_t _read_time; - int32_t _read_timezone; protected: uint8_t _endpoint; @@ -154,6 +125,10 @@ class ZigbeeEP { SemaphoreHandle_t lock; zb_power_source_t _power_source; + void addBoundDevice(zb_device_params_t *device) { + _bound_devices.push_back(device); + _is_bound = true; + } friend class ZigbeeCore; }; diff --git a/libraries/Zigbee/src/ZigbeeHandlers.cpp b/libraries/Zigbee/src/ZigbeeHandlers.cpp index 3af1b6c52aa..881d7ca0c37 100644 --- a/libraries/Zigbee/src/ZigbeeHandlers.cpp +++ b/libraries/Zigbee/src/ZigbeeHandlers.cpp @@ -4,37 +4,12 @@ #if SOC_IEEE802154_SUPPORTED && CONFIG_ZB_ENABLED -#include "esp_ota_ops.h" -#if CONFIG_ZB_DELTA_OTA // Delta OTA, code is prepared for this feature but not enabled by default -#include "esp_delta_ota_ops.h" -#endif - -//OTA Upgrade defines and variables -#define OTA_ELEMENT_HEADER_LEN 6 /* OTA element format header size include tag identifier and length field */ - -/** - * @name Enumeration for the tag identifier denotes the type and format of the data within the element - * @anchor esp_ota_element_tag_id_t - */ -typedef enum esp_ota_element_tag_id_e { - UPGRADE_IMAGE = 0x0000, /*!< Upgrade image */ -} esp_ota_element_tag_id_t; - -static const esp_partition_t *s_ota_partition = NULL; -static esp_ota_handle_t s_ota_handle = 0; -static bool s_tagid_received = false; - // forward declaration of all implemented handlers static esp_err_t zb_attribute_set_handler(const esp_zb_zcl_set_attr_value_message_t *message); static esp_err_t zb_attribute_reporting_handler(const esp_zb_zcl_report_attr_message_t *message); static esp_err_t zb_cmd_read_attr_resp_handler(const esp_zb_zcl_cmd_read_attr_resp_message_t *message); static esp_err_t zb_configure_report_resp_handler(const esp_zb_zcl_cmd_config_report_resp_message_t *message); -static esp_err_t zb_cmd_ias_zone_status_change_handler(const esp_zb_zcl_ias_zone_status_change_notification_message_t *message); -static esp_err_t zb_cmd_ias_zone_enroll_response_handler(const esp_zb_zcl_ias_zone_enroll_response_message_t *message); static esp_err_t zb_cmd_default_resp_handler(const esp_zb_zcl_cmd_default_resp_message_t *message); -static esp_err_t zb_window_covering_movement_resp_handler(const esp_zb_zcl_window_covering_movement_message_t *message); -static esp_err_t zb_ota_upgrade_status_handler(const esp_zb_zcl_ota_upgrade_value_message_t *message); -static esp_err_t zb_ota_upgrade_query_image_resp_handler(const esp_zb_zcl_ota_upgrade_query_image_resp_message_t *message); // Zigbee action handlers [[maybe_unused]] @@ -45,21 +20,8 @@ static esp_err_t zb_action_handler(esp_zb_core_action_callback_id_t callback_id, case ESP_ZB_CORE_REPORT_ATTR_CB_ID: ret = zb_attribute_reporting_handler((esp_zb_zcl_report_attr_message_t *)message); break; case ESP_ZB_CORE_CMD_READ_ATTR_RESP_CB_ID: ret = zb_cmd_read_attr_resp_handler((esp_zb_zcl_cmd_read_attr_resp_message_t *)message); break; case ESP_ZB_CORE_CMD_REPORT_CONFIG_RESP_CB_ID: ret = zb_configure_report_resp_handler((esp_zb_zcl_cmd_config_report_resp_message_t *)message); break; - case ESP_ZB_CORE_CMD_IAS_ZONE_ZONE_STATUS_CHANGE_NOT_ID: - ret = zb_cmd_ias_zone_status_change_handler((esp_zb_zcl_ias_zone_status_change_notification_message_t *)message); - break; - case ESP_ZB_CORE_IAS_ZONE_ENROLL_RESPONSE_VALUE_CB_ID: - ret = zb_cmd_ias_zone_enroll_response_handler((esp_zb_zcl_ias_zone_enroll_response_message_t *)message); - break; - case ESP_ZB_CORE_WINDOW_COVERING_MOVEMENT_CB_ID: - ret = zb_window_covering_movement_resp_handler((esp_zb_zcl_window_covering_movement_message_t *)message); - break; - case ESP_ZB_CORE_OTA_UPGRADE_VALUE_CB_ID: ret = zb_ota_upgrade_status_handler((esp_zb_zcl_ota_upgrade_value_message_t *)message); break; - case ESP_ZB_CORE_OTA_UPGRADE_QUERY_IMAGE_RESP_CB_ID: - ret = zb_ota_upgrade_query_image_resp_handler((esp_zb_zcl_ota_upgrade_query_image_resp_message_t *)message); - break; - case ESP_ZB_CORE_CMD_DEFAULT_RESP_CB_ID: ret = zb_cmd_default_resp_handler((esp_zb_zcl_cmd_default_resp_message_t *)message); break; - default: log_w("Receive unhandled Zigbee action(0x%x) callback", callback_id); break; + case ESP_ZB_CORE_CMD_DEFAULT_RESP_CB_ID: ret = zb_cmd_default_resp_handler((esp_zb_zcl_cmd_default_resp_message_t *)message); break; + default: log_w("Receive unhandled Zigbee action(0x%x) callback", callback_id); break; } return ret; } @@ -67,11 +29,9 @@ static esp_err_t zb_action_handler(esp_zb_core_action_callback_id_t callback_id, static esp_err_t zb_attribute_set_handler(const esp_zb_zcl_set_attr_value_message_t *message) { if (!message) { log_e("Empty message"); - return ESP_FAIL; } if (message->info.status != ESP_ZB_ZCL_STATUS_SUCCESS) { log_e("Received message: error status(%d)", message->info.status); - return ESP_ERR_INVALID_ARG; } log_v( @@ -95,11 +55,9 @@ static esp_err_t zb_attribute_set_handler(const esp_zb_zcl_set_attr_value_messag static esp_err_t zb_attribute_reporting_handler(const esp_zb_zcl_report_attr_message_t *message) { if (!message) { log_e("Empty message"); - return ESP_FAIL; } if (message->status != ESP_ZB_ZCL_STATUS_SUCCESS) { log_e("Received message: error status(%d)", message->status); - return ESP_ERR_INVALID_ARG; } log_v( "Received report from address(0x%x) src endpoint(%d) to dst endpoint(%d) cluster(0x%x)", message->src_address.u.short_addr, message->src_endpoint, @@ -117,11 +75,9 @@ static esp_err_t zb_attribute_reporting_handler(const esp_zb_zcl_report_attr_mes static esp_err_t zb_cmd_read_attr_resp_handler(const esp_zb_zcl_cmd_read_attr_resp_message_t *message) { if (!message) { log_e("Empty message"); - return ESP_FAIL; } if (message->info.status != ESP_ZB_ZCL_STATUS_SUCCESS) { log_e("Received message: error status(%d)", message->info.status); - return ESP_ERR_INVALID_ARG; } log_v( "Read attribute response: from address(0x%x) src endpoint(%d) to dst endpoint(%d) cluster(0x%x)", message->info.src_address.u.short_addr, @@ -139,8 +95,6 @@ static esp_err_t zb_cmd_read_attr_resp_handler(const esp_zb_zcl_cmd_read_attr_re if (variable->status == ESP_ZB_ZCL_STATUS_SUCCESS) { if (message->info.cluster == ESP_ZB_ZCL_CLUSTER_ID_BASIC) { (*it)->zbReadBasicCluster(&variable->attribute); //method zbReadBasicCluster implemented in the common EP class - } else if (message->info.cluster == ESP_ZB_ZCL_CLUSTER_ID_TIME) { - (*it)->zbReadTimeCluster(&variable->attribute); //method zbReadTimeCluster implemented in the common EP class } else { (*it)->zbAttributeRead(message->info.cluster, &variable->attribute); //method zbAttributeRead must be implemented in specific EP class } @@ -155,11 +109,9 @@ static esp_err_t zb_cmd_read_attr_resp_handler(const esp_zb_zcl_cmd_read_attr_re static esp_err_t zb_configure_report_resp_handler(const esp_zb_zcl_cmd_config_report_resp_message_t *message) { if (!message) { log_e("Empty message"); - return ESP_FAIL; } if (message->info.status != ESP_ZB_ZCL_STATUS_SUCCESS) { log_e("Received message: error status(%d)", message->info.status); - return ESP_ERR_INVALID_ARG; } esp_zb_zcl_config_report_resp_variable_t *variable = message->variables; while (variable) { @@ -172,223 +124,12 @@ static esp_err_t zb_configure_report_resp_handler(const esp_zb_zcl_cmd_config_re return ESP_OK; } -static esp_err_t zb_cmd_ias_zone_status_change_handler(const esp_zb_zcl_ias_zone_status_change_notification_message_t *message) { - if (!message) { - log_e("Empty message"); - return ESP_FAIL; - } - if (message->info.status != ESP_ZB_ZCL_STATUS_SUCCESS) { - log_e("Received message: error status(%d)", message->info.status); - return ESP_ERR_INVALID_ARG; - } - log_v( - "IAS Zone Status Notification: from address(0x%x) src endpoint(%d) to dst endpoint(%d) cluster(0x%x)", message->info.src_address.u.short_addr, - message->info.src_endpoint, message->info.dst_endpoint, message->info.cluster - ); - - for (std::list::iterator it = Zigbee.ep_objects.begin(); it != Zigbee.ep_objects.end(); ++it) { - if (message->info.dst_endpoint == (*it)->getEndpoint()) { - (*it)->zbIASZoneStatusChangeNotification(message); - } - } - return ESP_OK; -} - -static esp_err_t zb_cmd_ias_zone_enroll_response_handler(const esp_zb_zcl_ias_zone_enroll_response_message_t *message) { - if (!message) { - log_e("Empty message"); - return ESP_FAIL; - } - if (message->info.status != ESP_ZB_ZCL_STATUS_SUCCESS) { - log_e("Received message: error status(%d)", message->info.status); - return ESP_ERR_INVALID_ARG; - } - log_v("IAS Zone Enroll Response received"); - for (std::list::iterator it = Zigbee.ep_objects.begin(); it != Zigbee.ep_objects.end(); ++it) { - if (message->info.dst_endpoint == (*it)->getEndpoint()) { - (*it)->zbIASZoneEnrollResponse(message); - } - } - return ESP_OK; -} - -static esp_err_t zb_window_covering_movement_resp_handler(const esp_zb_zcl_window_covering_movement_message_t *message) { - if (!message) { - log_e("Empty message"); - } - if (message->info.status != ESP_ZB_ZCL_STATUS_SUCCESS) { - log_e("Received message: error status(%d)", message->info.status); - } - - log_v( - "Received message: endpoint(%d), cluster(0x%x), command(0x%x), payload(%d)", message->info.dst_endpoint, message->info.cluster, message->command, - message->payload - ); - - // List through all Zigbee EPs and call the callback function, with the message - for (std::list::iterator it = Zigbee.ep_objects.begin(); it != Zigbee.ep_objects.end(); ++it) { - if (message->info.dst_endpoint == (*it)->getEndpoint()) { - (*it)->zbWindowCoveringMovementCmd(message); //method zbWindowCoveringMovementCmd must be implemented in specific EP class - } - } - return ESP_OK; -} - -static esp_err_t esp_element_ota_data(uint32_t total_size, const void *payload, uint16_t payload_size, void **outbuf, uint16_t *outlen) { - static uint16_t tagid = 0; - void *data_buf = NULL; - uint16_t data_len; - - if (!s_tagid_received) { - uint32_t length = 0; - if (!payload || payload_size <= OTA_ELEMENT_HEADER_LEN) { - log_e("Invalid element format"); - return ESP_ERR_INVALID_ARG; - } - - const uint8_t *payload_ptr = (const uint8_t *)payload; - tagid = *(const uint16_t *)payload_ptr; - length = *(const uint32_t *)(payload_ptr + sizeof(tagid)); - if ((length + OTA_ELEMENT_HEADER_LEN) != total_size) { - log_e("Invalid element length [%ld/%ld]", length, total_size); - return ESP_ERR_INVALID_ARG; - } - - s_tagid_received = true; - - data_buf = (void *)(payload_ptr + OTA_ELEMENT_HEADER_LEN); - data_len = payload_size - OTA_ELEMENT_HEADER_LEN; - } else { - data_buf = (void *)payload; - data_len = payload_size; - } - - switch (tagid) { - case UPGRADE_IMAGE: - *outbuf = data_buf; - *outlen = data_len; - break; - default: - log_e("Unsupported element tag identifier %d", tagid); - return ESP_ERR_INVALID_ARG; - break; - } - - return ESP_OK; -} - -static esp_err_t zb_ota_upgrade_status_handler(const esp_zb_zcl_ota_upgrade_value_message_t *message) { - static uint32_t total_size = 0; - static uint32_t offset = 0; - [[maybe_unused]] - static int64_t start_time = 0; - esp_err_t ret = ESP_OK; - - if (message->info.status == ESP_ZB_ZCL_STATUS_SUCCESS) { - switch (message->upgrade_status) { - case ESP_ZB_ZCL_OTA_UPGRADE_STATUS_START: - log_i("Zigbee - OTA upgrade start"); - start_time = esp_timer_get_time(); - s_ota_partition = esp_ota_get_next_update_partition(NULL); - assert(s_ota_partition); -#if CONFIG_ZB_DELTA_OTA - ret = esp_delta_ota_begin(s_ota_partition, 0, &s_ota_handle); -#else - ret = esp_ota_begin(s_ota_partition, 0, &s_ota_handle); -#endif - if (ret != ESP_OK) { - log_e("Zigbee - Failed to begin OTA partition, status: %s", esp_err_to_name(ret)); - return ret; - } - break; - case ESP_ZB_ZCL_OTA_UPGRADE_STATUS_RECEIVE: - total_size = message->ota_header.image_size; - offset += message->payload_size; - log_i("Zigbee - OTA Client receives data: progress [%ld/%ld]", offset, total_size); - if (message->payload_size && message->payload) { - uint16_t payload_size = 0; - void *payload = NULL; - ret = esp_element_ota_data(total_size, message->payload, message->payload_size, &payload, &payload_size); - if (ret != ESP_OK) { - log_e("Zigbee - Failed to element OTA data, status: %s", esp_err_to_name(ret)); - return ret; - } -#if CONFIG_ZB_DELTA_OTA - ret = esp_delta_ota_write(s_ota_handle, payload, payload_size); -#else - ret = esp_ota_write(s_ota_handle, (const void *)payload, payload_size); -#endif - if (ret != ESP_OK) { - log_e("Zigbee - Failed to write OTA data to partition, status: %s", esp_err_to_name(ret)); - return ret; - } - } - break; - case ESP_ZB_ZCL_OTA_UPGRADE_STATUS_APPLY: log_i("Zigbee - OTA upgrade apply"); break; - case ESP_ZB_ZCL_OTA_UPGRADE_STATUS_CHECK: - ret = offset == total_size ? ESP_OK : ESP_FAIL; - offset = 0; - total_size = 0; - s_tagid_received = false; - log_i("Zigbee - OTA upgrade check status: %s", esp_err_to_name(ret)); - break; - case ESP_ZB_ZCL_OTA_UPGRADE_STATUS_FINISH: - log_i("Zigbee - OTA Finish"); - log_i( - "Zigbee - OTA Information: version: 0x%lx, manufacturer code: 0x%x, image type: 0x%x, total size: %ld bytes, cost time: %lld ms,", - message->ota_header.file_version, message->ota_header.manufacturer_code, message->ota_header.image_type, message->ota_header.image_size, - (esp_timer_get_time() - start_time) / 1000 - ); -#if CONFIG_ZB_DELTA_OTA - ret = esp_delta_ota_end(s_ota_handle); -#else - ret = esp_ota_end(s_ota_handle); -#endif - if (ret != ESP_OK) { - log_e("Zigbee - Failed to end OTA partition, status: %s", esp_err_to_name(ret)); - return ret; - } - ret = esp_ota_set_boot_partition(s_ota_partition); - if (ret != ESP_OK) { - log_e("Zigbee - Failed to set OTA boot partition, status: %s", esp_err_to_name(ret)); - return ret; - } - log_w("Zigbee - Prepare to restart system"); - esp_restart(); - break; - default: log_i("Zigbee - OTA status: %d", message->upgrade_status); break; - } - } - return ret; -} - -static esp_err_t zb_ota_upgrade_query_image_resp_handler(const esp_zb_zcl_ota_upgrade_query_image_resp_message_t *message) { - if (message->info.status == ESP_ZB_ZCL_STATUS_SUCCESS) { - log_i("Zigbee - Queried OTA image from address: 0x%04hx, endpoint: %d", message->server_addr.u.short_addr, message->server_endpoint); - log_i("Zigbee - Image version: 0x%lx, manufacturer code: 0x%x, image size: %ld", message->file_version, message->manufacturer_code, message->image_size); - if (message->image_size == 0) { - log_i("Zigbee - Rejecting OTA image upgrade, image size is 0"); - return ESP_FAIL; - } - if (message->file_version == 0) { - log_i("Zigbee - Rejecting OTA image upgrade, file version is 0"); - return ESP_FAIL; - } - log_i("Zigbee - Approving OTA image upgrade"); - } else { - log_i("Zigbee - OTA image upgrade response status: 0x%x", message->info.status); - } - return ESP_OK; -} - static esp_err_t zb_cmd_default_resp_handler(const esp_zb_zcl_cmd_default_resp_message_t *message) { if (!message) { log_e("Empty message"); - return ESP_FAIL; } if (message->info.status != ESP_ZB_ZCL_STATUS_SUCCESS) { log_e("Received message: error status(%d)", message->info.status); - return ESP_ERR_INVALID_ARG; } log_v( "Received default response: from address(0x%x), src_endpoint(%d) to dst_endpoint(%d), cluster(0x%x) with status 0x%x", diff --git a/libraries/Zigbee/src/ep/ZigbeeBasicOnOffSensor.cpp b/libraries/Zigbee/src/ep/ZigbeeBasicOnOffSensor.cpp new file mode 100644 index 00000000000..9726c254441 --- /dev/null +++ b/libraries/Zigbee/src/ep/ZigbeeBasicOnOffSensor.cpp @@ -0,0 +1,52 @@ +#include "ZigbeeBasicOnOffSensor.h" +#if SOC_IEEE802154_SUPPORTED && CONFIG_ZB_ENABLED + +esp_zb_cluster_list_t *zigbee_on_off_clusters_create(esp_zb_on_off_cfg_t *on_off_sensor) { + esp_zb_basic_cluster_cfg_t *basic_cfg = on_off_sensor ? &(on_off_sensor->basic_cfg) : NULL; + esp_zb_identify_cluster_cfg_t *identify_cfg = on_off_sensor ? &(on_off_sensor->identify_cfg) : NULL; + esp_zb_on_off_cluster_cfg_t *on_off_cfg = on_off_sensor ? &(on_off_sensor->state_cfg) : NULL; + esp_zb_cluster_list_t *cluster_list = esp_zb_zcl_cluster_list_create(); + esp_zb_cluster_list_add_basic_cluster(cluster_list, esp_zb_basic_cluster_create(basic_cfg), ESP_ZB_ZCL_CLUSTER_SERVER_ROLE); + esp_zb_cluster_list_add_identify_cluster(cluster_list, esp_zb_identify_cluster_create(identify_cfg), ESP_ZB_ZCL_CLUSTER_SERVER_ROLE); + esp_zb_cluster_list_add_on_off_cluster(cluster_list, esp_zb_on_off_cluster_create(on_off_cfg), ESP_ZB_ZCL_CLUSTER_SERVER_ROLE); + esp_zb_cluster_list_add_identify_cluster(cluster_list, esp_zb_zcl_attr_list_create(ESP_ZB_ZCL_CLUSTER_ID_IDENTIFY), ESP_ZB_ZCL_CLUSTER_SERVER_ROLE); + return cluster_list; +} + +ZigbeeBasicOnOffSensor::ZigbeeBasicOnOffSensor(uint8_t endpoint) : ZigbeeEP(endpoint) { + _device_id = ESP_ZB_HA_SIMPLE_SENSOR_DEVICE_ID; + + //Create custom occupancy sensor configuration + esp_zb_on_off_cfg_t on_off_cfg = ZIGBEE_DEFAULT_ON_OFF_CONFIG(); + _cluster_list = zigbee_on_off_clusters_create(&on_off_cfg); + + _ep_config = {.endpoint = _endpoint, .app_profile_id = ESP_ZB_AF_HA_PROFILE_ID, .app_device_id = ESP_ZB_HA_SIMPLE_SENSOR_DEVICE_ID, .app_device_version = 0}; +} + +void ZigbeeBasicOnOffSensor::setState(bool state) { + log_v("Updating on off sensor value..."); + /* Update on off sensor value */ + log_d("Setting state to %d", state); + esp_zb_lock_acquire(portMAX_DELAY); + esp_zb_zcl_set_attribute_val( + _endpoint, ESP_ZB_ZCL_CLUSTER_ID_ON_OFF, ESP_ZB_ZCL_CLUSTER_SERVER_ROLE, ESP_ZB_ZCL_ATTR_ON_OFF_ON_OFF_ID, &state, false + ); + esp_zb_lock_release(); +} + +void ZigbeeBasicOnOffSensor::report() { + /* Send report attributes command */ + esp_zb_zcl_report_attr_cmd_t report_attr_cmd; + report_attr_cmd.address_mode = ESP_ZB_APS_ADDR_MODE_DST_ADDR_ENDP_NOT_PRESENT; + report_attr_cmd.attributeID = ESP_ZB_ZCL_ATTR_ON_OFF_ON_OFF_ID; + report_attr_cmd.direction = ESP_ZB_ZCL_CMD_DIRECTION_TO_CLI; + report_attr_cmd.clusterID = ESP_ZB_ZCL_CLUSTER_ID_ON_OFF; + report_attr_cmd.zcl_basic_cmd.src_endpoint = _endpoint; + + esp_zb_lock_acquire(portMAX_DELAY); + esp_zb_zcl_report_attr_cmd_req(&report_attr_cmd); + esp_zb_lock_release(); + log_v("State report sent"); +} + +#endif //SOC_IEEE802154_SUPPORTED && CONFIG_ZB_ENABLED diff --git a/libraries/Zigbee/src/ep/ZigbeeBasicOnOffSensor.h b/libraries/Zigbee/src/ep/ZigbeeBasicOnOffSensor.h new file mode 100644 index 00000000000..38ccfc128e1 --- /dev/null +++ b/libraries/Zigbee/src/ep/ZigbeeBasicOnOffSensor.h @@ -0,0 +1,48 @@ +/* Class of Zigbee Pressure sensor endpoint inherited from common EP class */ + +#pragma once + +#include "soc/soc_caps.h" +#include "sdkconfig.h" +#if SOC_IEEE802154_SUPPORTED && CONFIG_ZB_ENABLED + +#include "ZigbeeEP.h" +#include "ha/esp_zigbee_ha_standard.h" + +// clang-format off +#define ZIGBEE_DEFAULT_ON_OFF_CONFIG() \ + { \ + .basic_cfg = \ + { \ + .zcl_version = ESP_ZB_ZCL_BASIC_ZCL_VERSION_DEFAULT_VALUE, \ + .power_source = ESP_ZB_ZCL_BASIC_POWER_SOURCE_DEFAULT_VALUE, \ + }, \ + .identify_cfg = \ + { \ + .identify_time = ESP_ZB_ZCL_IDENTIFY_IDENTIFY_TIME_DEFAULT_VALUE, \ + }, \ + .state_cfg = { \ + .on_off = ESP_ZB_ZCL_ON_OFF_ON_OFF_DEFAULT_VALUE, \ + } \ + } \ +// clang-format on + +typedef struct esp_zb_on_off_cfg_s { + esp_zb_basic_cluster_cfg_t basic_cfg; + esp_zb_identify_cluster_cfg_t identify_cfg; + esp_zb_on_off_cluster_cfg_t state_cfg; +} esp_zb_on_off_cfg_t; + +class ZigbeeBasicOnOffSensor : public ZigbeeEP { +public: + ZigbeeBasicOnOffSensor(uint8_t endpoint); + ~ZigbeeBasicOnOffSensor(); + + // Set the occupancy value. True for occupied, false for unoccupied + void setState(bool state); + + // Report the occupancy value + void report(); +}; + +#endif //SOC_IEEE802154_SUPPORTED && CONFIG_ZB_ENABLED diff --git a/libraries/Zigbee/src/ep/ZigbeeCarbonDioxideSensor.cpp b/libraries/Zigbee/src/ep/ZigbeeCarbonDioxideSensor.cpp index b4386b80386..7e966550e90 100644 --- a/libraries/Zigbee/src/ep/ZigbeeCarbonDioxideSensor.cpp +++ b/libraries/Zigbee/src/ep/ZigbeeCarbonDioxideSensor.cpp @@ -11,6 +11,7 @@ esp_zb_cluster_list_t *zigbee_carbon_dioxide_sensor_clusters_create(zigbee_carbo esp_zb_cluster_list_add_carbon_dioxide_measurement_cluster( cluster_list, esp_zb_carbon_dioxide_measurement_cluster_create(carbon_dioxide_meas_cfg), ESP_ZB_ZCL_CLUSTER_SERVER_ROLE ); + esp_zb_cluster_list_add_identify_cluster(cluster_list, esp_zb_zcl_attr_list_create(ESP_ZB_ZCL_CLUSTER_ID_IDENTIFY), ESP_ZB_ZCL_CLUSTER_SERVER_ROLE); return cluster_list; } @@ -37,7 +38,7 @@ void ZigbeeCarbonDioxideSensor::setTolerance(float tolerance) { float zb_tolerance = tolerance / 1000000.0f; esp_zb_attribute_list_t *carbon_dioxide_measure_cluster = esp_zb_cluster_list_get_cluster(_cluster_list, ESP_ZB_ZCL_CLUSTER_ID_CARBON_DIOXIDE_MEASUREMENT, ESP_ZB_ZCL_CLUSTER_SERVER_ROLE); - esp_zb_temperature_meas_cluster_add_attr(carbon_dioxide_measure_cluster, ESP_ZB_ZCL_ATTR_CARBON_DIOXIDE_MEASUREMENT_TOLERANCE_ID, (void *)&zb_tolerance); + esp_zb_carbon_dioxide_measurement_cluster_add_attr(carbon_dioxide_measure_cluster, ESP_ZB_ZCL_ATTR_CARBON_DIOXIDE_MEASUREMENT_TOLERANCE_ID, (void *)&zb_tolerance); } void ZigbeeCarbonDioxideSensor::setReporting(uint16_t min_interval, uint16_t max_interval, uint16_t delta) { @@ -55,11 +56,9 @@ void ZigbeeCarbonDioxideSensor::setReporting(uint16_t min_interval, uint16_t max reporting_info.u.send_info.max_interval = max_interval; reporting_info.u.send_info.def_min_interval = min_interval; reporting_info.u.send_info.def_max_interval = max_interval; + reporting_info.u.send_info.delta.u16 = delta; reporting_info.dst.profile_id = ESP_ZB_AF_HA_PROFILE_ID; reporting_info.manuf_code = ESP_ZB_ZCL_ATTR_NON_MANUFACTURER_SPECIFIC; - float delta_f = delta / 1000000.0f; - memcpy(&reporting_info.u.send_info.delta.s32, &delta_f, sizeof(float)); - esp_zb_lock_acquire(portMAX_DELAY); esp_zb_zcl_update_reporting_info(&reporting_info); esp_zb_lock_release(); diff --git a/libraries/Zigbee/src/ep/ZigbeeCarbonDioxideSensor.h b/libraries/Zigbee/src/ep/ZigbeeCarbonDioxideSensor.h index 7744fd02f00..71c353d4695 100644 --- a/libraries/Zigbee/src/ep/ZigbeeCarbonDioxideSensor.h +++ b/libraries/Zigbee/src/ep/ZigbeeCarbonDioxideSensor.h @@ -39,7 +39,7 @@ typedef struct zigbee_carbon_dioxide_sensor_cfg_s { class ZigbeeCarbonDioxideSensor : public ZigbeeEP { public: ZigbeeCarbonDioxideSensor(uint8_t endpoint); - ~ZigbeeCarbonDioxideSensor() {} + ~ZigbeeCarbonDioxideSensor(); // Set the carbon dioxide value in ppm void setCarbonDioxide(float carbon_dioxide); diff --git a/libraries/Zigbee/src/ep/ZigbeeColorDimmableLight.cpp b/libraries/Zigbee/src/ep/ZigbeeColorDimmableLight.cpp index 7ffd6976e1f..f034daba54a 100644 --- a/libraries/Zigbee/src/ep/ZigbeeColorDimmableLight.cpp +++ b/libraries/Zigbee/src/ep/ZigbeeColorDimmableLight.cpp @@ -4,17 +4,8 @@ ZigbeeColorDimmableLight::ZigbeeColorDimmableLight(uint8_t endpoint) : ZigbeeEP(endpoint) { _device_id = ESP_ZB_HA_COLOR_DIMMABLE_LIGHT_DEVICE_ID; - esp_zb_color_dimmable_light_cfg_t light_cfg = ZIGBEE_DEFAULT_COLOR_DIMMABLE_LIGHT_CONFIG(); + esp_zb_color_dimmable_light_cfg_t light_cfg = ESP_ZB_DEFAULT_COLOR_DIMMABLE_LIGHT_CONFIG(); _cluster_list = esp_zb_color_dimmable_light_clusters_create(&light_cfg); - - //Add support for hue and saturation - uint8_t hue = 0; - uint8_t saturation = 0; - - esp_zb_attribute_list_t *color_cluster = esp_zb_cluster_list_get_cluster(_cluster_list, ESP_ZB_ZCL_CLUSTER_ID_COLOR_CONTROL, ESP_ZB_ZCL_CLUSTER_SERVER_ROLE); - esp_zb_color_control_cluster_add_attr(color_cluster, ESP_ZB_ZCL_ATTR_COLOR_CONTROL_CURRENT_HUE_ID, &hue); - esp_zb_color_control_cluster_add_attr(color_cluster, ESP_ZB_ZCL_ATTR_COLOR_CONTROL_CURRENT_SATURATION_ID, &saturation); - _ep_config = { .endpoint = _endpoint, .app_profile_id = ESP_ZB_AF_HA_PROFILE_ID, .app_device_id = ESP_ZB_HA_COLOR_DIMMABLE_LIGHT_DEVICE_ID, .app_device_version = 0 }; @@ -22,7 +13,9 @@ ZigbeeColorDimmableLight::ZigbeeColorDimmableLight(uint8_t endpoint) : ZigbeeEP( //set default values _current_state = false; _current_level = 255; - _current_color = {255, 255, 255}; + _current_red = 255; + _current_green = 255; + _current_blue = 255; } uint16_t ZigbeeColorDimmableLight::getCurrentColorX() { @@ -39,18 +32,37 @@ uint16_t ZigbeeColorDimmableLight::getCurrentColorY() { ->data_p); } -uint8_t ZigbeeColorDimmableLight::getCurrentColorHue() { - return (*(uint8_t *)esp_zb_zcl_get_attribute( - _endpoint, ESP_ZB_ZCL_CLUSTER_ID_COLOR_CONTROL, ESP_ZB_ZCL_CLUSTER_SERVER_ROLE, ESP_ZB_ZCL_ATTR_COLOR_CONTROL_CURRENT_HUE_ID - ) - ->data_p); +void ZigbeeColorDimmableLight::calculateRGB(uint16_t x, uint16_t y, uint8_t &red, uint8_t &green, uint8_t &blue) { + float r, g, b, color_x, color_y; + color_x = (float)x / 65535; + color_y = (float)y / 65535; + + float color_X = color_x / color_y; + float color_Z = (1 - color_x - color_y) / color_y; + + XYZ_TO_RGB(color_X, 1, color_Z, r, g, b); + + red = (uint8_t)(r * (float)255); + green = (uint8_t)(g * (float)255); + blue = (uint8_t)(b * (float)255); } -uint8_t ZigbeeColorDimmableLight::getCurrentColorSaturation() { - return (*(uint16_t *)esp_zb_zcl_get_attribute( - _endpoint, ESP_ZB_ZCL_CLUSTER_ID_COLOR_CONTROL, ESP_ZB_ZCL_CLUSTER_SERVER_ROLE, ESP_ZB_ZCL_ATTR_COLOR_CONTROL_CURRENT_SATURATION_ID - ) - ->data_p); +void ZigbeeColorDimmableLight::calculateXY(uint8_t red, uint8_t green, uint8_t blue, uint16_t &x, uint16_t &y) { + // Convert RGB to XYZ + float r = (float)red / 255.0f; + float g = (float)green / 255.0f; + float b = (float)blue / 255.0f; + + float X, Y, Z; + RGB_TO_XYZ(r, g, b, X, Y, Z); + + // Convert XYZ to xy chromaticity coordinates + float color_x = X / (X + Y + Z); + float color_y = Y / (X + Y + Z); + + // Convert normalized xy to 16-bit values + x = (uint16_t)(color_x * 65535.0f); + y = (uint16_t)(color_y * 65535.0f); } //set attribute method -> method overridden in child class @@ -82,7 +94,11 @@ void ZigbeeColorDimmableLight::zbAttributeSet(const esp_zb_zcl_set_attr_value_me uint16_t light_color_x = (*(uint16_t *)message->attribute.data.value); uint16_t light_color_y = getCurrentColorY(); //calculate RGB from XY and call setColor() - _current_color = espXYToRgbColor(255, light_color_x, light_color_y); //TODO: Check if level is correct + uint8_t red, green, blue; + calculateRGB(light_color_x, light_color_y, red, green, blue); + _current_blue = blue; + _current_green = green; + _current_red = red; lightChanged(); return; @@ -90,17 +106,11 @@ void ZigbeeColorDimmableLight::zbAttributeSet(const esp_zb_zcl_set_attr_value_me uint16_t light_color_x = getCurrentColorX(); uint16_t light_color_y = (*(uint16_t *)message->attribute.data.value); //calculate RGB from XY and call setColor() - _current_color = espXYToRgbColor(255, light_color_x, light_color_y); //TODO: Check if level is correct - lightChanged(); - return; - } else if (message->attribute.id == ESP_ZB_ZCL_ATTR_COLOR_CONTROL_CURRENT_HUE_ID && message->attribute.data.type == ESP_ZB_ZCL_ATTR_TYPE_U8) { - uint8_t light_color_hue = (*(uint8_t *)message->attribute.data.value); - _current_color = espHsvToRgbColor(light_color_hue, getCurrentColorSaturation(), 255); - lightChanged(); - return; - } else if (message->attribute.id == ESP_ZB_ZCL_ATTR_COLOR_CONTROL_CURRENT_SATURATION_ID && message->attribute.data.type == ESP_ZB_ZCL_ATTR_TYPE_U8) { - uint8_t light_color_saturation = (*(uint8_t *)message->attribute.data.value); - _current_color = espHsvToRgbColor(getCurrentColorHue(), light_color_saturation, 255); + uint8_t red, green, blue; + calculateRGB(light_color_x, light_color_y, red, green, blue); + _current_blue = blue; + _current_green = green; + _current_red = red; lightChanged(); return; } else { @@ -113,7 +123,7 @@ void ZigbeeColorDimmableLight::zbAttributeSet(const esp_zb_zcl_set_attr_value_me void ZigbeeColorDimmableLight::lightChanged() { if (_on_light_change) { - _on_light_change(_current_state, _current_color.r, _current_color.g, _current_color.b, _current_level); + _on_light_change(_current_state, _current_red, _current_green, _current_blue, _current_level); } } @@ -121,13 +131,12 @@ void ZigbeeColorDimmableLight::setLight(bool state, uint8_t level, uint8_t red, //Update all attributes _current_state = state; _current_level = level; - _current_color = {red, green, blue}; + _current_red = red; + _current_green = green; + _current_blue = blue; lightChanged(); - espXyColor_t xy_color = espRgbColorToXYColor(_current_color); - espHsvColor_t hsv_color = espRgbColorToHsvColor(_current_color); - - log_v("Updating light state: %d, level: %d, color: %d, %d, %d", state, level, red, green, blue); + log_v("Updating on/off light state to %d", state); /* Update light clusters */ esp_zb_lock_acquire(portMAX_DELAY); //set on/off state @@ -138,43 +147,28 @@ void ZigbeeColorDimmableLight::setLight(bool state, uint8_t level, uint8_t red, esp_zb_zcl_set_attribute_val( _endpoint, ESP_ZB_ZCL_CLUSTER_ID_LEVEL_CONTROL, ESP_ZB_ZCL_CLUSTER_SERVER_ROLE, ESP_ZB_ZCL_ATTR_LEVEL_CONTROL_CURRENT_LEVEL_ID, &_current_level, false ); - //set xy color - esp_zb_zcl_set_attribute_val( - _endpoint, ESP_ZB_ZCL_CLUSTER_ID_COLOR_CONTROL, ESP_ZB_ZCL_CLUSTER_SERVER_ROLE, ESP_ZB_ZCL_ATTR_COLOR_CONTROL_CURRENT_X_ID, &xy_color.x, false - ); + //set color + uint16_t color_x, color_y; + calculateXY(red, green, blue, color_x, color_y); esp_zb_zcl_set_attribute_val( - _endpoint, ESP_ZB_ZCL_CLUSTER_ID_COLOR_CONTROL, ESP_ZB_ZCL_CLUSTER_SERVER_ROLE, ESP_ZB_ZCL_ATTR_COLOR_CONTROL_CURRENT_Y_ID, &xy_color.y, false + _endpoint, ESP_ZB_ZCL_CLUSTER_ID_COLOR_CONTROL, ESP_ZB_ZCL_CLUSTER_SERVER_ROLE, ESP_ZB_ZCL_ATTR_COLOR_CONTROL_CURRENT_X_ID, &color_x, false ); - //set hsv color - uint8_t hue = (uint8_t)hsv_color.h; esp_zb_zcl_set_attribute_val( - _endpoint, ESP_ZB_ZCL_CLUSTER_ID_COLOR_CONTROL, ESP_ZB_ZCL_CLUSTER_SERVER_ROLE, ESP_ZB_ZCL_ATTR_COLOR_CONTROL_CURRENT_HUE_ID, &hue, false - ); - esp_zb_zcl_set_attribute_val( - _endpoint, ESP_ZB_ZCL_CLUSTER_ID_COLOR_CONTROL, ESP_ZB_ZCL_CLUSTER_SERVER_ROLE, ESP_ZB_ZCL_ATTR_COLOR_CONTROL_CURRENT_SATURATION_ID, &hsv_color.s, false + _endpoint, ESP_ZB_ZCL_CLUSTER_ID_COLOR_CONTROL, ESP_ZB_ZCL_CLUSTER_SERVER_ROLE, ESP_ZB_ZCL_ATTR_COLOR_CONTROL_CURRENT_Y_ID, &color_y, false ); esp_zb_lock_release(); } void ZigbeeColorDimmableLight::setLightState(bool state) { - setLight(state, _current_level, _current_color.r, _current_color.g, _current_color.b); + setLight(state, _current_level, _current_red, _current_green, _current_blue); } void ZigbeeColorDimmableLight::setLightLevel(uint8_t level) { - setLight(_current_state, level, _current_color.r, _current_color.g, _current_color.b); + setLight(_current_state, level, _current_red, _current_green, _current_blue); } void ZigbeeColorDimmableLight::setLightColor(uint8_t red, uint8_t green, uint8_t blue) { setLight(_current_state, _current_level, red, green, blue); } -void ZigbeeColorDimmableLight::setLightColor(espRgbColor_t rgb_color) { - setLight(_current_state, _current_level, rgb_color.r, rgb_color.g, rgb_color.b); -} - -void ZigbeeColorDimmableLight::setLightColor(espHsvColor_t hsv_color) { - espRgbColor_t rgb_color = espHsvColorToRgbColor(hsv_color); - setLight(_current_state, _current_level, rgb_color.r, rgb_color.g, rgb_color.b); -} - #endif //SOC_IEEE802154_SUPPORTED && CONFIG_ZB_ENABLED diff --git a/libraries/Zigbee/src/ep/ZigbeeColorDimmableLight.h b/libraries/Zigbee/src/ep/ZigbeeColorDimmableLight.h index 265fec1b37c..9fa59dcfffc 100644 --- a/libraries/Zigbee/src/ep/ZigbeeColorDimmableLight.h +++ b/libraries/Zigbee/src/ep/ZigbeeColorDimmableLight.h @@ -9,51 +9,10 @@ #include "ZigbeeEP.h" #include "ha/esp_zigbee_ha_standard.h" -#define ZIGBEE_DEFAULT_COLOR_DIMMABLE_LIGHT_CONFIG() \ - { \ - .basic_cfg = \ - { \ - .zcl_version = ESP_ZB_ZCL_BASIC_ZCL_VERSION_DEFAULT_VALUE, \ - .power_source = ESP_ZB_ZCL_BASIC_POWER_SOURCE_DEFAULT_VALUE, \ - }, \ - .identify_cfg = \ - { \ - .identify_time = ESP_ZB_ZCL_IDENTIFY_IDENTIFY_TIME_DEFAULT_VALUE, \ - }, \ - .groups_cfg = \ - { \ - .groups_name_support_id = ESP_ZB_ZCL_GROUPS_NAME_SUPPORT_DEFAULT_VALUE, \ - }, \ - .scenes_cfg = \ - { \ - .scenes_count = ESP_ZB_ZCL_SCENES_SCENE_COUNT_DEFAULT_VALUE, \ - .current_scene = ESP_ZB_ZCL_SCENES_CURRENT_SCENE_DEFAULT_VALUE, \ - .current_group = ESP_ZB_ZCL_SCENES_CURRENT_GROUP_DEFAULT_VALUE, \ - .scene_valid = ESP_ZB_ZCL_SCENES_SCENE_VALID_DEFAULT_VALUE, \ - .name_support = ESP_ZB_ZCL_SCENES_NAME_SUPPORT_DEFAULT_VALUE, \ - }, \ - .on_off_cfg = \ - { \ - .on_off = ESP_ZB_ZCL_ON_OFF_ON_OFF_DEFAULT_VALUE, \ - }, \ - .level_cfg = \ - { \ - .current_level = ESP_ZB_ZCL_LEVEL_CONTROL_CURRENT_LEVEL_DEFAULT_VALUE, \ - }, \ - .color_cfg = { \ - .current_x = ESP_ZB_ZCL_COLOR_CONTROL_CURRENT_X_DEF_VALUE, \ - .current_y = ESP_ZB_ZCL_COLOR_CONTROL_CURRENT_Y_DEF_VALUE, \ - .color_mode = ESP_ZB_ZCL_COLOR_CONTROL_COLOR_MODE_DEFAULT_VALUE, \ - .options = ESP_ZB_ZCL_COLOR_CONTROL_OPTIONS_DEFAULT_VALUE, \ - .enhanced_color_mode = ESP_ZB_ZCL_COLOR_CONTROL_ENHANCED_COLOR_MODE_DEFAULT_VALUE, \ - .color_capabilities = 0x0009, \ - }, \ - } - class ZigbeeColorDimmableLight : public ZigbeeEP { public: ZigbeeColorDimmableLight(uint8_t endpoint); - ~ZigbeeColorDimmableLight() {} + ~ZigbeeColorDimmableLight(); void onLightChange(void (*callback)(bool, uint8_t, uint8_t, uint8_t, uint8_t)) { _on_light_change = callback; @@ -65,8 +24,6 @@ class ZigbeeColorDimmableLight : public ZigbeeEP { void setLightState(bool state); void setLightLevel(uint8_t level); void setLightColor(uint8_t red, uint8_t green, uint8_t blue); - void setLightColor(espRgbColor_t rgb_color); - void setLightColor(espHsvColor_t hsv_color); void setLight(bool state, uint8_t level, uint8_t red, uint8_t green, uint8_t blue); bool getLightState() { @@ -75,26 +32,23 @@ class ZigbeeColorDimmableLight : public ZigbeeEP { uint8_t getLightLevel() { return _current_level; } - espRgbColor_t getLightColor() { - return _current_color; - } uint8_t getLightRed() { - return _current_color.r; + return _current_red; } uint8_t getLightGreen() { - return _current_color.g; + return _current_green; } uint8_t getLightBlue() { - return _current_color.b; + return _current_blue; } private: void zbAttributeSet(const esp_zb_zcl_set_attr_value_message_t *message) override; + void calculateRGB(uint16_t x, uint16_t y, uint8_t &red, uint8_t &green, uint8_t &blue); + void calculateXY(uint8_t red, uint8_t green, uint8_t blue, uint16_t &x, uint16_t &y); uint16_t getCurrentColorX(); uint16_t getCurrentColorY(); - uint8_t getCurrentColorHue(); - uint8_t getCurrentColorSaturation(); void lightChanged(); //callback function to be called on light change (State, R, G, B, Level) @@ -102,7 +56,9 @@ class ZigbeeColorDimmableLight : public ZigbeeEP { bool _current_state; uint8_t _current_level; - espRgbColor_t _current_color; + uint16_t _current_red; + uint16_t _current_green; + uint16_t _current_blue; }; #endif //SOC_IEEE802154_SUPPORTED && CONFIG_ZB_ENABLED diff --git a/libraries/Zigbee/src/ep/ZigbeeColorDimmerSwitch.cpp b/libraries/Zigbee/src/ep/ZigbeeColorDimmerSwitch.cpp index 7bdd8b8ad6a..4fd492a5477 100644 --- a/libraries/Zigbee/src/ep/ZigbeeColorDimmerSwitch.cpp +++ b/libraries/Zigbee/src/ep/ZigbeeColorDimmerSwitch.cpp @@ -16,6 +16,24 @@ ZigbeeColorDimmerSwitch::ZigbeeColorDimmerSwitch(uint8_t endpoint) : ZigbeeEP(en }; } +void ZigbeeColorDimmerSwitch::calculateXY(uint8_t red, uint8_t green, uint8_t blue, uint16_t &x, uint16_t &y) { + // Convert RGB to XYZ + float r = (float)red / 255.0f; + float g = (float)green / 255.0f; + float b = (float)blue / 255.0f; + + float X, Y, Z; + RGB_TO_XYZ(r, g, b, X, Y, Z); + + // Convert XYZ to xy chromaticity coordinates + float color_x = X / (X + Y + Z); + float color_y = Y / (X + Y + Z); + + // Convert normalized xy to 16-bit values + x = (uint16_t)(color_x * 65535.0f); + y = (uint16_t)(color_y * 65535.0f); +} + void ZigbeeColorDimmerSwitch::bindCb(esp_zb_zdp_status_t zdo_status, void *user_ctx) { if (zdo_status == ESP_ZB_ZDP_STATUS_SUCCESS) { log_i("Bound successfully!"); @@ -399,13 +417,15 @@ void ZigbeeColorDimmerSwitch::setLightLevel(uint8_t level, uint8_t endpoint, esp void ZigbeeColorDimmerSwitch::setLightColor(uint8_t red, uint8_t green, uint8_t blue) { if (_is_bound) { - espXyColor_t xy_color = espRgbToXYColor(red, green, blue); + //Convert RGB to XY + uint16_t color_x, color_y; + calculateXY(red, green, blue, color_x, color_y); esp_zb_zcl_color_move_to_color_cmd_t cmd_req; cmd_req.zcl_basic_cmd.src_endpoint = _endpoint; cmd_req.address_mode = ESP_ZB_APS_ADDR_MODE_DST_ADDR_ENDP_NOT_PRESENT; - cmd_req.color_x = xy_color.x; - cmd_req.color_y = xy_color.y; + cmd_req.color_x = color_x; + cmd_req.color_y = color_y; cmd_req.transition_time = 0; log_v("Sending 'set light color' command"); esp_zb_lock_acquire(portMAX_DELAY); @@ -418,14 +438,16 @@ void ZigbeeColorDimmerSwitch::setLightColor(uint8_t red, uint8_t green, uint8_t void ZigbeeColorDimmerSwitch::setLightColor(uint8_t red, uint8_t green, uint8_t blue, uint16_t group_addr) { if (_is_bound) { - espXyColor_t xy_color = espRgbToXYColor(red, green, blue); + //Convert RGB to XY + uint16_t color_x, color_y; + calculateXY(red, green, blue, color_x, color_y); esp_zb_zcl_color_move_to_color_cmd_t cmd_req; cmd_req.zcl_basic_cmd.src_endpoint = _endpoint; cmd_req.zcl_basic_cmd.dst_addr_u.addr_short = group_addr; cmd_req.address_mode = ESP_ZB_APS_ADDR_MODE_16_GROUP_ENDP_NOT_PRESENT; - cmd_req.color_x = xy_color.x; - cmd_req.color_y = xy_color.y; + cmd_req.color_x = color_x; + cmd_req.color_y = color_y; cmd_req.transition_time = 0; log_v("Sending 'set light color' command to group address 0x%x", group_addr); esp_zb_lock_acquire(portMAX_DELAY); @@ -438,15 +460,17 @@ void ZigbeeColorDimmerSwitch::setLightColor(uint8_t red, uint8_t green, uint8_t void ZigbeeColorDimmerSwitch::setLightColor(uint8_t red, uint8_t green, uint8_t blue, uint8_t endpoint, uint16_t short_addr) { if (_is_bound) { - espXyColor_t xy_color = espRgbToXYColor(red, green, blue); + //Convert RGB to XY + uint16_t color_x, color_y; + calculateXY(red, green, blue, color_x, color_y); esp_zb_zcl_color_move_to_color_cmd_t cmd_req; cmd_req.zcl_basic_cmd.src_endpoint = _endpoint; cmd_req.zcl_basic_cmd.dst_endpoint = endpoint; cmd_req.zcl_basic_cmd.dst_addr_u.addr_short = short_addr; cmd_req.address_mode = ESP_ZB_APS_ADDR_MODE_16_ENDP_PRESENT; - cmd_req.color_x = xy_color.x; - cmd_req.color_y = xy_color.y; + cmd_req.color_x = color_x; + cmd_req.color_y = color_y; cmd_req.transition_time = 0; log_v("Sending 'set light color' command to endpoint %d, address 0x%x", endpoint, short_addr); esp_zb_lock_acquire(portMAX_DELAY); @@ -459,15 +483,17 @@ void ZigbeeColorDimmerSwitch::setLightColor(uint8_t red, uint8_t green, uint8_t void ZigbeeColorDimmerSwitch::setLightColor(uint8_t red, uint8_t green, uint8_t blue, uint8_t endpoint, esp_zb_ieee_addr_t ieee_addr) { if (_is_bound) { - espXyColor_t xy_color = espRgbToXYColor(red, green, blue); + //Convert RGB to XY + uint16_t color_x, color_y; + calculateXY(red, green, blue, color_x, color_y); esp_zb_zcl_color_move_to_color_cmd_t cmd_req; cmd_req.zcl_basic_cmd.src_endpoint = _endpoint; cmd_req.zcl_basic_cmd.dst_endpoint = endpoint; cmd_req.address_mode = ESP_ZB_APS_ADDR_MODE_64_ENDP_PRESENT; memcpy(cmd_req.zcl_basic_cmd.dst_addr_u.addr_long, ieee_addr, sizeof(esp_zb_ieee_addr_t)); - cmd_req.color_x = xy_color.x; - cmd_req.color_y = xy_color.y; + cmd_req.color_x = color_x; + cmd_req.color_y = color_y; cmd_req.transition_time = 0; log_v( "Sending 'set light color' command to endpoint %d, ieee address %02x:%02x:%02x:%02x:%02x:%02x:%02x:%02x", endpoint, ieee_addr[7], ieee_addr[6], diff --git a/libraries/Zigbee/src/ep/ZigbeeColorDimmerSwitch.h b/libraries/Zigbee/src/ep/ZigbeeColorDimmerSwitch.h index ca67fb4ba62..8e2a4d9e1a3 100644 --- a/libraries/Zigbee/src/ep/ZigbeeColorDimmerSwitch.h +++ b/libraries/Zigbee/src/ep/ZigbeeColorDimmerSwitch.h @@ -12,7 +12,7 @@ class ZigbeeColorDimmerSwitch : public ZigbeeEP { public: ZigbeeColorDimmerSwitch(uint8_t endpoint); - ~ZigbeeColorDimmerSwitch() {} + ~ZigbeeColorDimmerSwitch(); // methods to control the color dimmable light void lightToggle(); diff --git a/libraries/Zigbee/src/ep/ZigbeeDimmableLight.h b/libraries/Zigbee/src/ep/ZigbeeDimmableLight.h index 6f92c3315e4..034c34899b4 100644 --- a/libraries/Zigbee/src/ep/ZigbeeDimmableLight.h +++ b/libraries/Zigbee/src/ep/ZigbeeDimmableLight.h @@ -67,7 +67,7 @@ typedef struct zigbee_dimmable_light_cfg_s { class ZigbeeDimmableLight : public ZigbeeEP { public: ZigbeeDimmableLight(uint8_t endpoint); - ~ZigbeeDimmableLight() {} + ~ZigbeeDimmableLight(); void onLightChange(void (*callback)(bool, uint8_t)) { _on_light_change = callback; diff --git a/libraries/Zigbee/src/ep/ZigbeeFlowSensor.cpp b/libraries/Zigbee/src/ep/ZigbeeFlowSensor.cpp index 36d66840967..75196e78543 100644 --- a/libraries/Zigbee/src/ep/ZigbeeFlowSensor.cpp +++ b/libraries/Zigbee/src/ep/ZigbeeFlowSensor.cpp @@ -9,6 +9,7 @@ esp_zb_cluster_list_t *zigbee_flow_sensor_clusters_create(zigbee_flow_sensor_cfg esp_zb_cluster_list_add_basic_cluster(cluster_list, esp_zb_basic_cluster_create(basic_cfg), ESP_ZB_ZCL_CLUSTER_SERVER_ROLE); esp_zb_cluster_list_add_identify_cluster(cluster_list, esp_zb_identify_cluster_create(identify_cfg), ESP_ZB_ZCL_CLUSTER_SERVER_ROLE); esp_zb_cluster_list_add_flow_meas_cluster(cluster_list, esp_zb_flow_meas_cluster_create(flow_meas_cfg), ESP_ZB_ZCL_CLUSTER_SERVER_ROLE); + esp_zb_cluster_list_add_identify_cluster(cluster_list, esp_zb_zcl_attr_list_create(ESP_ZB_ZCL_CLUSTER_ID_IDENTIFY), ESP_ZB_ZCL_CLUSTER_SERVER_ROLE); return cluster_list; } diff --git a/libraries/Zigbee/src/ep/ZigbeeFlowSensor.h b/libraries/Zigbee/src/ep/ZigbeeFlowSensor.h index 7d5ec26f7ec..3514e2fcc1b 100644 --- a/libraries/Zigbee/src/ep/ZigbeeFlowSensor.h +++ b/libraries/Zigbee/src/ep/ZigbeeFlowSensor.h @@ -39,7 +39,7 @@ typedef struct zigbee_flow_sensor_cfg_s { class ZigbeeFlowSensor : public ZigbeeEP { public: ZigbeeFlowSensor(uint8_t endpoint); - ~ZigbeeFlowSensor() {} + ~ZigbeeFlowSensor(); // Set the flow value in 0,1 m3/h void setFlow(float value); diff --git a/libraries/Zigbee/src/ep/ZigbeeLight.h b/libraries/Zigbee/src/ep/ZigbeeLight.h index 8cf8c35f781..9b8fc409d4a 100644 --- a/libraries/Zigbee/src/ep/ZigbeeLight.h +++ b/libraries/Zigbee/src/ep/ZigbeeLight.h @@ -12,7 +12,7 @@ class ZigbeeLight : public ZigbeeEP { public: ZigbeeLight(uint8_t endpoint); - ~ZigbeeLight() {} + ~ZigbeeLight(); // Use to set a cb function to be called on light change void onLightChange(void (*callback)(bool)) { diff --git a/libraries/Zigbee/src/ep/ZigbeeLightSensor.cpp b/libraries/Zigbee/src/ep/ZigbeeLightSensor.cpp new file mode 100644 index 00000000000..e8f9232159e --- /dev/null +++ b/libraries/Zigbee/src/ep/ZigbeeLightSensor.cpp @@ -0,0 +1,76 @@ +#include "ZigbeeLightSensor.h" +#if SOC_IEEE802154_SUPPORTED && CONFIG_ZB_ENABLED + +ZigbeeLightSensor::ZigbeeLightSensor(uint8_t endpoint) : ZigbeeEP(endpoint) { + _device_id = ESP_ZB_HA_LIGHT_SENSOR_DEVICE_ID; + + esp_zb_light_sensor_cfg_t light_sensor_cfg = ESP_ZB_DEFAULT_LIGHT_SENSOR_CONFIG(); + _cluster_list = esp_zb_light_sensor_clusters_create(&light_sensor_cfg); + + _ep_config = { + .endpoint = _endpoint, .app_profile_id = ESP_ZB_AF_HA_PROFILE_ID, .app_device_id = ESP_ZB_HA_LIGHT_SENSOR_DEVICE_ID, .app_device_version = 0 + }; +} + + + +void ZigbeeLightSensor::setMinMaxValue(uint16_t zb_min, uint16_t zb_max) { + esp_zb_attribute_list_t *light_measure_cluster = + esp_zb_cluster_list_get_cluster(_cluster_list, ESP_ZB_ZCL_CLUSTER_ID_ILLUMINANCE_MEASUREMENT, ESP_ZB_ZCL_CLUSTER_SERVER_ROLE); + esp_zb_cluster_update_attr(light_measure_cluster, ESP_ZB_ZCL_ATTR_ILLUMINANCE_MEASUREMENT_MIN_MEASURED_VALUE_ID, (void *)&zb_min); + esp_zb_cluster_update_attr(light_measure_cluster, ESP_ZB_ZCL_ATTR_ILLUMINANCE_MEASUREMENT_MAX_MEASURED_VALUE_ID, (void *)&zb_max); +} + +void ZigbeeLightSensor::setTolerance(uint16_t zb_tolerance) { + esp_zb_attribute_list_t *light_measure_cluster = + esp_zb_cluster_list_get_cluster(_cluster_list, ESP_ZB_ZCL_CLUSTER_ID_ILLUMINANCE_MEASUREMENT, ESP_ZB_ZCL_CLUSTER_SERVER_ROLE); + esp_zb_illuminance_meas_cluster_add_attr(light_measure_cluster, ESP_ZB_ZCL_ATTR_ILLUMINANCE_MEASUREMENT_TOLERANCE_ID, (void *)&zb_tolerance); +} + +void ZigbeeLightSensor::setReporting(uint16_t min_interval, uint16_t max_interval, float delta) { + esp_zb_zcl_reporting_info_t reporting_info; + memset(&reporting_info, 0, sizeof(esp_zb_zcl_reporting_info_t)); + reporting_info.direction = ESP_ZB_ZCL_CMD_DIRECTION_TO_SRV; + reporting_info.ep = _endpoint; + reporting_info.cluster_id = ESP_ZB_ZCL_CLUSTER_ID_ILLUMINANCE_MEASUREMENT; + reporting_info.cluster_role = ESP_ZB_ZCL_CLUSTER_SERVER_ROLE; + reporting_info.attr_id = ESP_ZB_ZCL_ATTR_ILLUMINANCE_MEASUREMENT_MEASURED_VALUE_ID; + reporting_info.u.send_info.min_interval = min_interval; + reporting_info.u.send_info.max_interval = max_interval; + reporting_info.u.send_info.def_min_interval = min_interval; + reporting_info.u.send_info.def_max_interval = max_interval; + reporting_info.u.send_info.delta.u16 = (uint16_t)(delta * 100); + reporting_info.dst.profile_id = ESP_ZB_AF_HA_PROFILE_ID; + reporting_info.manuf_code = ESP_ZB_ZCL_ATTR_NON_MANUFACTURER_SPECIFIC, esp_zb_lock_acquire(portMAX_DELAY); + esp_zb_zcl_update_reporting_info(&reporting_info); + esp_zb_lock_release(); +} + +void ZigbeeLightSensor::setIlluminance(uint16_t value) { + log_v("Updating light sensor value..."); + /* Update light sensor measured value */ + log_d("Setting illuminance to %d", value); + esp_zb_lock_acquire(portMAX_DELAY); + esp_zb_zcl_set_attribute_val( + _endpoint, ESP_ZB_ZCL_CLUSTER_ID_ILLUMINANCE_MEASUREMENT, ESP_ZB_ZCL_CLUSTER_SERVER_ROLE, ESP_ZB_ZCL_ATTR_ILLUMINANCE_MEASUREMENT_MEASURED_VALUE_ID, &value, false + ); + esp_zb_lock_release(); +} + +void ZigbeeLightSensor::reportIlluminance() { + /* Send report attributes command */ + esp_zb_zcl_report_attr_cmd_t report_attr_cmd; + report_attr_cmd.address_mode = ESP_ZB_APS_ADDR_MODE_DST_ADDR_ENDP_NOT_PRESENT; + report_attr_cmd.attributeID = ESP_ZB_ZCL_ATTR_ILLUMINANCE_MEASUREMENT_MEASURED_VALUE_ID; + report_attr_cmd.direction = ESP_ZB_ZCL_CMD_DIRECTION_TO_CLI; + report_attr_cmd.clusterID = ESP_ZB_ZCL_CLUSTER_ID_ILLUMINANCE_MEASUREMENT; + report_attr_cmd.zcl_basic_cmd.src_endpoint = _endpoint; + + esp_zb_lock_acquire(portMAX_DELAY); + esp_zb_zcl_report_attr_cmd_req(&report_attr_cmd); + esp_zb_lock_release(); + log_v("Illuminance report sent"); +} + + +#endif //SOC_IEEE802154_SUPPORTED && CONFIG_ZB_ENABLED diff --git a/libraries/Zigbee/src/ep/ZigbeeLightSensor.h b/libraries/Zigbee/src/ep/ZigbeeLightSensor.h new file mode 100644 index 00000000000..fa79fd2e305 --- /dev/null +++ b/libraries/Zigbee/src/ep/ZigbeeLightSensor.h @@ -0,0 +1,53 @@ +/* Class of Zigbee Light Sensor sensor endpoint inherited from common EP class */ + +#pragma once + +#include "soc/soc_caps.h" +#include "sdkconfig.h" +#if SOC_IEEE802154_SUPPORTED && CONFIG_ZB_ENABLED + +#include "ZigbeeEP.h" +#include "ha/esp_zigbee_ha_standard.h" + +#define ESP_ZB_DEFAULT_LIGHT_SENSOR_CONFIG() \ + { \ + .basic_cfg = \ + { \ + .zcl_version = ESP_ZB_ZCL_BASIC_ZCL_VERSION_DEFAULT_VALUE, \ + .power_source = ESP_ZB_ZCL_BASIC_POWER_SOURCE_DEFAULT_VALUE, \ + }, \ + .identify_cfg = \ + { \ + .identify_time = ESP_ZB_ZCL_IDENTIFY_IDENTIFY_TIME_DEFAULT_VALUE, \ + }, \ + .illuminance_cfg = \ + { \ + .measured_value = ESP_ZB_ZCL_ILLUMINANCE_MEASUREMENT_LIGHT_SENSOR_TYPE_DEFAULT_VALUE, \ + .min_value = ESP_ZB_ZCL_ILLUMINANCE_MEASUREMENT_LIGHT_SENSOR_TYPE_DEFAULT_VALUE, \ + .max_value = ESP_ZB_ZCL_ILLUMINANCE_MEASUREMENT_LIGHT_SENSOR_TYPE_DEFAULT_VALUE, \ + }, \ + } / + +class ZigbeeLightSensor : public ZigbeeEP { +public: + ZigbeeLightSensor(uint8_t endpoint); + ~ZigbeeLightSensor(); + + // Set the light value in lux + void setIlluminance(uint16_t value); + + // Set the min and max value for the light sensor in lux + void setMinMaxValue(uint16_t min, uint16_t max); + + // Set the tolerance value of the light sensor + void setTolerance(uint16_t zb_tolerance); + + // Set the reporting interval for light measurement and delta in lux + void setReporting(uint16_t min_interval, uint16_t max_interval, float delta); + + // Report the light value + void reportIlluminance(); + +}; + +#endif //SOC_IEEE802154_SUPPORTED && CONFIG_ZB_ENABLED diff --git a/libraries/Zigbee/src/ep/ZigbeeOccupancySensor.cpp b/libraries/Zigbee/src/ep/ZigbeeOccupancySensor.cpp index dec7910ac03..3a7acee040c 100644 --- a/libraries/Zigbee/src/ep/ZigbeeOccupancySensor.cpp +++ b/libraries/Zigbee/src/ep/ZigbeeOccupancySensor.cpp @@ -9,6 +9,7 @@ esp_zb_cluster_list_t *zigbee_occupancy_sensor_clusters_create(zigbee_occupancy_ esp_zb_cluster_list_add_basic_cluster(cluster_list, esp_zb_basic_cluster_create(basic_cfg), ESP_ZB_ZCL_CLUSTER_SERVER_ROLE); esp_zb_cluster_list_add_identify_cluster(cluster_list, esp_zb_identify_cluster_create(identify_cfg), ESP_ZB_ZCL_CLUSTER_SERVER_ROLE); esp_zb_cluster_list_add_occupancy_sensing_cluster(cluster_list, esp_zb_occupancy_sensing_cluster_create(occupancy_meas_cfg), ESP_ZB_ZCL_CLUSTER_SERVER_ROLE); + esp_zb_cluster_list_add_identify_cluster(cluster_list, esp_zb_zcl_attr_list_create(ESP_ZB_ZCL_CLUSTER_ID_IDENTIFY), ESP_ZB_ZCL_CLUSTER_SERVER_ROLE); return cluster_list; } diff --git a/libraries/Zigbee/src/ep/ZigbeeOccupancySensor.h b/libraries/Zigbee/src/ep/ZigbeeOccupancySensor.h index 40c5eddbbdd..e4b25c063e5 100644 --- a/libraries/Zigbee/src/ep/ZigbeeOccupancySensor.h +++ b/libraries/Zigbee/src/ep/ZigbeeOccupancySensor.h @@ -39,7 +39,7 @@ typedef struct zigbee_occupancy_sensor_cfg_s { class ZigbeeOccupancySensor : public ZigbeeEP { public: ZigbeeOccupancySensor(uint8_t endpoint); - ~ZigbeeOccupancySensor() {} + ~ZigbeeOccupancySensor(); // Set the occupancy value. True for occupied, false for unoccupied void setOccupancy(bool occupied); diff --git a/libraries/Zigbee/src/ep/ZigbeePressureSensor.cpp b/libraries/Zigbee/src/ep/ZigbeePressureSensor.cpp index b887680076d..f41be7c5a17 100644 --- a/libraries/Zigbee/src/ep/ZigbeePressureSensor.cpp +++ b/libraries/Zigbee/src/ep/ZigbeePressureSensor.cpp @@ -9,6 +9,7 @@ esp_zb_cluster_list_t *zigbee_pressure_sensor_clusters_create(zigbee_pressure_se esp_zb_cluster_list_add_basic_cluster(cluster_list, esp_zb_basic_cluster_create(basic_cfg), ESP_ZB_ZCL_CLUSTER_SERVER_ROLE); esp_zb_cluster_list_add_identify_cluster(cluster_list, esp_zb_identify_cluster_create(identify_cfg), ESP_ZB_ZCL_CLUSTER_SERVER_ROLE); esp_zb_cluster_list_add_pressure_meas_cluster(cluster_list, esp_zb_pressure_meas_cluster_create(pressure_meas_cfg), ESP_ZB_ZCL_CLUSTER_SERVER_ROLE); + esp_zb_cluster_list_add_identify_cluster(cluster_list, esp_zb_zcl_attr_list_create(ESP_ZB_ZCL_CLUSTER_ID_IDENTIFY), ESP_ZB_ZCL_CLUSTER_SERVER_ROLE); return cluster_list; } @@ -33,7 +34,7 @@ void ZigbeePressureSensor::setMinMaxValue(int16_t min, int16_t max) { void ZigbeePressureSensor::setTolerance(uint16_t tolerance) { esp_zb_attribute_list_t *pressure_measure_cluster = esp_zb_cluster_list_get_cluster(_cluster_list, ESP_ZB_ZCL_CLUSTER_ID_PRESSURE_MEASUREMENT, ESP_ZB_ZCL_CLUSTER_SERVER_ROLE); - esp_zb_temperature_meas_cluster_add_attr(pressure_measure_cluster, ESP_ZB_ZCL_ATTR_PRESSURE_MEASUREMENT_TOLERANCE_ID, (void *)&tolerance); + esp_zb_pressure_meas_cluster_add_attr(pressure_measure_cluster, ESP_ZB_ZCL_ATTR_PRESSURE_MEASUREMENT_TOLERANCE_ID, (void *)&tolerance); } void ZigbeePressureSensor::setReporting(uint16_t min_interval, uint16_t max_interval, uint16_t delta) { diff --git a/libraries/Zigbee/src/ep/ZigbeePressureSensor.h b/libraries/Zigbee/src/ep/ZigbeePressureSensor.h index 2d72ef04c06..f088c033bb9 100644 --- a/libraries/Zigbee/src/ep/ZigbeePressureSensor.h +++ b/libraries/Zigbee/src/ep/ZigbeePressureSensor.h @@ -39,7 +39,7 @@ typedef struct zigbee_pressure_sensor_cfg_s { class ZigbeePressureSensor : public ZigbeeEP { public: ZigbeePressureSensor(uint8_t endpoint); - ~ZigbeePressureSensor() {} + ~ZigbeePressureSensor(); // Set the pressure value in 1 hPa void setPressure(int16_t value); diff --git a/libraries/Zigbee/src/ep/ZigbeeSwitch.h b/libraries/Zigbee/src/ep/ZigbeeSwitch.h index b638bfe823a..62264641378 100644 --- a/libraries/Zigbee/src/ep/ZigbeeSwitch.h +++ b/libraries/Zigbee/src/ep/ZigbeeSwitch.h @@ -12,7 +12,7 @@ class ZigbeeSwitch : public ZigbeeEP { public: ZigbeeSwitch(uint8_t endpoint); - ~ZigbeeSwitch() {} + ~ZigbeeSwitch(); // methods to control the on/off light void lightToggle(); diff --git a/libraries/Zigbee/src/ep/ZigbeeTempSensor.h b/libraries/Zigbee/src/ep/ZigbeeTempSensor.h index e610ff8d356..2951d4b7628 100644 --- a/libraries/Zigbee/src/ep/ZigbeeTempSensor.h +++ b/libraries/Zigbee/src/ep/ZigbeeTempSensor.h @@ -12,7 +12,7 @@ class ZigbeeTempSensor : public ZigbeeEP { public: ZigbeeTempSensor(uint8_t endpoint); - ~ZigbeeTempSensor() {} + ~ZigbeeTempSensor(); // Set the temperature value in 0,01°C void setTemperature(float value); diff --git a/libraries/Zigbee/src/ep/ZigbeeThermostat.h b/libraries/Zigbee/src/ep/ZigbeeThermostat.h index 669ed9ab50a..fe797ffd7b6 100644 --- a/libraries/Zigbee/src/ep/ZigbeeThermostat.h +++ b/libraries/Zigbee/src/ep/ZigbeeThermostat.h @@ -32,7 +32,7 @@ class ZigbeeThermostat : public ZigbeeEP { public: ZigbeeThermostat(uint8_t endpoint); - ~ZigbeeThermostat() {} + ~ZigbeeThermostat(); void onTempRecieve(void (*callback)(float)) { _on_temp_recieve = callback; From 40d6de265b389070ad57b95e90307c36e5b17808 Mon Sep 17 00:00:00 2001 From: Andrew Date: Wed, 19 Feb 2025 21:31:18 +0000 Subject: [PATCH 2/3] Fixed readmes --- .../examples/Zigbee_Basic_On_Off/README.md | 4 ++-- .../examples/Zigbee_Light_Sensor/README.md | 18 ++++++------------ 2 files changed, 8 insertions(+), 14 deletions(-) diff --git a/libraries/Zigbee/examples/Zigbee_Basic_On_Off/README.md b/libraries/Zigbee/examples/Zigbee_Basic_On_Off/README.md index 0c5dcd013f2..5b80f56114f 100644 --- a/libraries/Zigbee/examples/Zigbee_Basic_On_Off/README.md +++ b/libraries/Zigbee/examples/Zigbee_Basic_On_Off/README.md @@ -1,6 +1,6 @@ -# Arduino-ESP32 Zigbee Occupancy Sensor Example +# Arduino-ESP32 Zigbee Basic On Off Example -This example shows how to configure the Zigbee end device and use it as a Home Automation (HA) occupancy sensor (PIR). +This example shows how to configure the Zigbee end device and use it as a Home Automation (HA) basic on off switch. # Supported Targets diff --git a/libraries/Zigbee/examples/Zigbee_Light_Sensor/README.md b/libraries/Zigbee/examples/Zigbee_Light_Sensor/README.md index 577bd7c8058..6fa9d485ee6 100644 --- a/libraries/Zigbee/examples/Zigbee_Light_Sensor/README.md +++ b/libraries/Zigbee/examples/Zigbee_Light_Sensor/README.md @@ -1,6 +1,6 @@ -# Arduino-ESP32 Zigbee Temperature Sensor Example +# Arduino-ESP32 Zigbee Light Sensor Example -This example shows how to configure the Zigbee end device and use it as a Home Automation (HA) temperature sensor. +This example shows how to configure the Zigbee end device and use it as a Home Automation (HA) light sensor. # Supported Targets @@ -9,25 +9,19 @@ Currently, this example supports the following targets. | Supported Targets | ESP32-C6 | ESP32-H2 | | ----------------- | -------- | -------- | -## Temperature Sensor Functions - -Note: - * This board means the board (e.g. ESP32-H2 / C6) loaded with `Zigbee_Temperature_Sensor` example. - * The remote board means the board (e.g. ESP32-H2 / C6) loaded with `Zigbee_Thermostat` example. +## Light Sensor Functions Functions: - * After this board first starts up, it would be configured locally to report the temperature on 1 degree change and no periodic reporting to the remote board. - * By clicking the button (BOOT) on this board, this board will immediately send a report of the current measured temperature to the remote board. + * After this board first starts up, it would be configured locally to report the illuminance on 1 lux change and no periodic reporting to the remote board. + * By clicking the button (BOOT) on this board, this board will immediately send a report of the current measured illuminance to the remote board. ## Hardware Required -* One development board (ESP32-H2 or ESP32-C6) acting as Zigbee coordinator (loaded with `Zigbee_Thermostat` example) * A USB cable for power supply and programming -* Choose another board (ESP32-H2 or ESP32-C6) as Zigbee end device and upload the `Zigbee_Temperature_Sensor` example ### Configure the Project -In this example, the internal temperature sensor task is reading the chip temperature. +In this example, the internal temperature sensor task is reading the chip temperature - replace this with your light sensor readings. Set the Button GPIO by changing the `button` variable. By default, it's the pin `BOOT_PIN` (BOOT button on ESP32-C6 and ESP32-H2). #### Using Arduino IDE From d794402b6cfd15be4dd56ccfec7f0b3f401eaf66 Mon Sep 17 00:00:00 2001 From: Andrew Date: Thu, 20 Feb 2025 16:42:05 +0000 Subject: [PATCH 3/3] Split on off sensor and device, fixed light sensor default config --- .../Zigbee_Basic_On_Off.ino | 32 +++-- .../Zigbee_Basic_On_Off_Sensor/README.md | 57 +++++++++ .../Zigbee_Basic_On_Off_Sensor.ino | 109 ++++++++++++++++++ .../Zigbee_Basic_On_Off_Sensor/ci.json | 6 + libraries/Zigbee/src/Zigbee.h | 1 + .../Zigbee/src/ep/ZigbeeBasicOnOffDevice.cpp | 77 +++++++++++++ .../Zigbee/src/ep/ZigbeeBasicOnOffDevice.h | 62 ++++++++++ .../Zigbee/src/ep/ZigbeeBasicOnOffSensor.cpp | 10 +- .../Zigbee/src/ep/ZigbeeBasicOnOffSensor.h | 9 +- libraries/Zigbee/src/ep/ZigbeeLightSensor.h | 2 +- 10 files changed, 343 insertions(+), 22 deletions(-) create mode 100644 libraries/Zigbee/examples/Zigbee_Basic_On_Off_Sensor/README.md create mode 100644 libraries/Zigbee/examples/Zigbee_Basic_On_Off_Sensor/Zigbee_Basic_On_Off_Sensor.ino create mode 100644 libraries/Zigbee/examples/Zigbee_Basic_On_Off_Sensor/ci.json create mode 100644 libraries/Zigbee/src/ep/ZigbeeBasicOnOffDevice.cpp create mode 100644 libraries/Zigbee/src/ep/ZigbeeBasicOnOffDevice.h diff --git a/libraries/Zigbee/examples/Zigbee_Basic_On_Off/Zigbee_Basic_On_Off.ino b/libraries/Zigbee/examples/Zigbee_Basic_On_Off/Zigbee_Basic_On_Off.ino index 405e9518c10..c78ce4ebad8 100644 --- a/libraries/Zigbee/examples/Zigbee_Basic_On_Off/Zigbee_Basic_On_Off.ino +++ b/libraries/Zigbee/examples/Zigbee_Basic_On_Off/Zigbee_Basic_On_Off.ino @@ -13,18 +13,19 @@ // limitations under the License. /** - * @brief This example demonstrates Zigbee occupancy sensor. + * @brief This example demonstrates Zigbee light sensor. * - * The example demonstrates how to use Zigbee library to create a end device occupancy sensor. - * The occupancy sensor is a Zigbee end device, which is reporting data to the Zigbee network. - * Tested with PIR sensor HC-SR501 connected to GPIO4. + * The example demonstrates how to use Zigbee library to create an end device basic on of device. + * The on off device is a Zigbee end device, which is reporting data to the Zigbee network and + * receiving state back. * * Proper Zigbee mode must be selected in Tools->Zigbee mode * and also the correct partition scheme must be selected in Tools->Partition Scheme. * * Please check the README.md for instructions and more detailed description. * - * Created by Jan Procházka (https://github.com/P-R-O-C-H-Y/) + * By Andrew McDonald (https://github.com/andymanic) + * Modfied from code by Jan Procházka (https://github.com/P-R-O-C-H-Y/) */ #ifndef ZIGBEE_MODE_ED @@ -38,7 +39,11 @@ uint8_t button = BOOT_PIN; uint8_t sensor_pin = 4; -ZigbeeBasicOnOffSensor zbOnOffSensor = ZigbeeBasicOnOffSensor(ON_OFF_SENSOR_ENDPOINT_NUMBER); +ZigbeeBasicOnOffDevice zbOnOffDevice = ZigbeeBasicOnOffDevice(ON_OFF_SENSOR_ENDPOINT_NUMBER); + +void setState(bool value) { + digitalWrite(LED_BUILTIN, value); +} void setup() { Serial.begin(115200); @@ -48,10 +53,13 @@ void setup() { pinMode(sensor_pin, INPUT); // Optional: set Zigbee device name and model - zbOnOffSensor.setManufacturerAndModel("Espressif", "ZigbeeOnOffSensor"); + zbOnOffDevice.setManufacturerAndModel("Espressif", "ZigbeeOnOffDevice"); + + // Set callback function for receiving state changes from Zigbee + zbOnOffDevice.onStateChange(setState); // Add endpoint to Zigbee Core - Zigbee.addEndpoint(&zbOnOffSensor); + Zigbee.addEndpoint(&zbOnOffDevice); Serial.println("Starting Zigbee..."); // When all EPs are registered, start Zigbee in End Device mode @@ -75,12 +83,12 @@ void loop() { static bool state = false; if (digitalRead(sensor_pin) == HIGH && !state) { // Update state value - zbOnOffSensor.setState(true); - zbOnOffSensor.report(); + zbOnOffDevice.setState(true); + zbOnOffDevice.report(); state = true; } else if (digitalRead(sensor_pin) == LOW && state) { - zbOnOffSensor.setState(false); - zbOnOffSensor.report(); + zbOnOffDevice.setState(false); + zbOnOffDevice.report(); state = false; } diff --git a/libraries/Zigbee/examples/Zigbee_Basic_On_Off_Sensor/README.md b/libraries/Zigbee/examples/Zigbee_Basic_On_Off_Sensor/README.md new file mode 100644 index 00000000000..5b80f56114f --- /dev/null +++ b/libraries/Zigbee/examples/Zigbee_Basic_On_Off_Sensor/README.md @@ -0,0 +1,57 @@ +# Arduino-ESP32 Zigbee Basic On Off Example + +This example shows how to configure the Zigbee end device and use it as a Home Automation (HA) basic on off switch. + +# Supported Targets + +Currently, this example supports the following targets. + +| Supported Targets | ESP32-C6 | ESP32-H2 | +| ----------------- | -------- | -------- | + +## Hardware Required + +* A USB cable for power supply and programming + +### Configure the Project + +Set the Button GPIO by changing the `button` variable. By default, it's the pin `BOOT_PIN` (BOOT button on ESP32-C6 and ESP32-H2). +Set the Sensor GPIO by changing the `sensor_pin` variable. + +#### Using Arduino IDE + +To get more information about the Espressif boards see [Espressif Development Kits](https://www.espressif.com/en/products/devkits). + +* Before Compile/Verify, select the correct board: `Tools -> Board`. +* Select the End device Zigbee mode: `Tools -> Zigbee mode: Zigbee ED (end device)` +* Select Partition Scheme for Zigbee: `Tools -> Partition Scheme: Zigbee 4MB with spiffs` +* Select the COM port: `Tools -> Port: xxx` where the `xxx` is the detected COM port. +* Optional: Set debug level to verbose to see all logs from Zigbee stack: `Tools -> Core Debug Level: Verbose`. + +## Troubleshooting + +If the End device flashed with this example is not connecting to the coordinator, erase the flash of the End device before flashing the example to the board. + +***Important: Make sure you are using a good quality USB cable and that you have a reliable power source*** + +* **LED not blinking:** Check the wiring connection and the IO selection. +* **Programming Fail:** If the programming/flash procedure fails, try reducing the serial connection speed. +* **COM port not detected:** Check the USB cable and the USB to Serial driver installation. + +If the error persists, you can ask for help at the official [ESP32 forum](https://esp32.com) or see [Contribute](#contribute). + +## Contribute + +To know how to contribute to this project, see [How to contribute.](https://github.com/espressif/arduino-esp32/blob/master/CONTRIBUTING.rst) + +If you have any **feedback** or **issue** to report on this example/library, please open an issue or fix it by creating a new PR. Contributions are more than welcome! + +Before creating a new issue, be sure to try Troubleshooting and check if the same issue was already created by someone else. + +## Resources + +* Official ESP32 Forum: [Link](https://esp32.com) +* Arduino-ESP32 Official Repository: [espressif/arduino-esp32](https://github.com/espressif/arduino-esp32) +* ESP32-C6 Datasheet: [Link to datasheet](https://www.espressif.com/sites/default/files/documentation/esp32-c6_datasheet_en.pdf) +* ESP32-H2 Datasheet: [Link to datasheet](https://www.espressif.com/sites/default/files/documentation/esp32-h2_datasheet_en.pdf) +* Official ESP-IDF documentation: [ESP-IDF](https://idf.espressif.com) diff --git a/libraries/Zigbee/examples/Zigbee_Basic_On_Off_Sensor/Zigbee_Basic_On_Off_Sensor.ino b/libraries/Zigbee/examples/Zigbee_Basic_On_Off_Sensor/Zigbee_Basic_On_Off_Sensor.ino new file mode 100644 index 00000000000..a3f9d495288 --- /dev/null +++ b/libraries/Zigbee/examples/Zigbee_Basic_On_Off_Sensor/Zigbee_Basic_On_Off_Sensor.ino @@ -0,0 +1,109 @@ +// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +/** + * @brief This example demonstrates Zigbee light sensor. + * + * The example demonstrates how to use Zigbee library to create an end device basic on of device. + * The on off device is a Zigbee end device, which is reporting data to the Zigbee network and + * receiving state back. + * + * Proper Zigbee mode must be selected in Tools->Zigbee mode + * and also the correct partition scheme must be selected in Tools->Partition Scheme. + * + * Please check the README.md for instructions and more detailed description. + * + * By Andrew McDonald (https://github.com/andymanic) + * Modfied from code by Jan Procházka (https://github.com/P-R-O-C-H-Y/) + */ + +#ifndef ZIGBEE_MODE_ED +#error "Zigbee end device mode is not selected in Tools->Zigbee mode" +#endif + +#include "Zigbee.h" + +/* Zigbee on off sensor configuration */ +#define ON_OFF_SENSOR_ENDPOINT_NUMBER 10 +uint8_t button = BOOT_PIN; +uint8_t sensor_pin = 4; +uint8_t led = LED_BUILTIN; + +ZigbeeBasicOnOffSensor zbOnOffDevice = ZigbeeBasicOnOffSensor(ON_OFF_SENSOR_ENDPOINT_NUMBER); + +void setup() { + Serial.begin(115200); + + // Init button + PIR sensor + pinMode(button, INPUT_PULLUP); + pinMode(sensor_pin, INPUT); + + pinMode(led, OUTPUT); + digitalWrite(led, LOW); + // Optional: set Zigbee device name and model + zbOnOffDevice.setManufacturerAndModel("Espressif", "ZigbeeOnOffSensor"); + + // Add endpoint to Zigbee Core + Zigbee.addEndpoint(&zbOnOffDevice); + + Serial.println("Starting Zigbee..."); + // When all EPs are registered, start Zigbee in End Device mode + if (!Zigbee.begin()) { + Serial.println("Zigbee failed to start!"); + Serial.println("Rebooting..."); + ESP.restart(); + } else { + Serial.println("Zigbee started successfully!"); + } + Serial.println("Connecting to network"); + while (!Zigbee.connected()) { + Serial.print("."); + delay(100); + } + Serial.println(); +} + +void loop() { + // Check input state + static bool state = false; + if (digitalRead(sensor_pin) == HIGH && !state) { + // Update state value + zbOnOffDevice.setState(true); + digitalWrite(led, HIGH); + zbOnOffDevice.report(); + state = true; + } else if (digitalRead(sensor_pin) == LOW && state) { + zbOnOffDevice.setState(false); + digitalWrite(led, LOW); + zbOnOffDevice.report(); + state = false; + } + + // Checking button for factory reset + if (digitalRead(button) == LOW) { // Push button pressed + // Key debounce handling + delay(100); + int startTime = millis(); + while (digitalRead(button) == LOW) { + delay(50); + if ((millis() - startTime) > 3000) { + // If key pressed for more than 3secs, factory reset Zigbee and reboot + Serial.println("Resetting Zigbee to factory and rebooting in 1s."); + delay(1000); + Zigbee.factoryReset(); + } + } + } + delay(100); +} diff --git a/libraries/Zigbee/examples/Zigbee_Basic_On_Off_Sensor/ci.json b/libraries/Zigbee/examples/Zigbee_Basic_On_Off_Sensor/ci.json new file mode 100644 index 00000000000..7b7ccef8ed7 --- /dev/null +++ b/libraries/Zigbee/examples/Zigbee_Basic_On_Off_Sensor/ci.json @@ -0,0 +1,6 @@ +{ + "fqbn_append": "PartitionScheme=zigbee,ZigbeeMode=ed", + "requires": [ + "CONFIG_SOC_IEEE802154_SUPPORTED=y" + ] +} diff --git a/libraries/Zigbee/src/Zigbee.h b/libraries/Zigbee/src/Zigbee.h index 700d0f2243a..a9cf67830de 100644 --- a/libraries/Zigbee/src/Zigbee.h +++ b/libraries/Zigbee/src/Zigbee.h @@ -20,3 +20,4 @@ #include "ep/ZigbeeCarbonDioxideSensor.h" #include "ep/ZigbeeLightSensor.h" #include "ep/ZigbeeBasicOnOffSensor.h" +#include "ep/ZigbeeBasicOnOffDevice.h" diff --git a/libraries/Zigbee/src/ep/ZigbeeBasicOnOffDevice.cpp b/libraries/Zigbee/src/ep/ZigbeeBasicOnOffDevice.cpp new file mode 100644 index 00000000000..f3106990e81 --- /dev/null +++ b/libraries/Zigbee/src/ep/ZigbeeBasicOnOffDevice.cpp @@ -0,0 +1,77 @@ +#include "ZigbeeBasicOnOffDevice.h" +#if SOC_IEEE802154_SUPPORTED && CONFIG_ZB_ENABLED + +esp_zb_cluster_list_t *zigbee_on_off_clusters_create(esp_zb_on_off_cfg_t *on_off_sensor) { + esp_zb_basic_cluster_cfg_t *basic_cfg = on_off_sensor ? &(on_off_sensor->basic_cfg) : NULL; + esp_zb_identify_cluster_cfg_t *identify_cfg = on_off_sensor ? &(on_off_sensor->identify_cfg) : NULL; + esp_zb_on_off_cluster_cfg_t *on_off_cfg = on_off_sensor ? &(on_off_sensor->state_cfg) : NULL; + esp_zb_cluster_list_t *cluster_list = esp_zb_zcl_cluster_list_create(); + esp_zb_cluster_list_add_basic_cluster(cluster_list, esp_zb_basic_cluster_create(basic_cfg), ESP_ZB_ZCL_CLUSTER_SERVER_ROLE); + esp_zb_cluster_list_add_identify_cluster(cluster_list, esp_zb_identify_cluster_create(identify_cfg), ESP_ZB_ZCL_CLUSTER_SERVER_ROLE); + esp_zb_cluster_list_add_on_off_cluster(cluster_list, esp_zb_on_off_cluster_create(on_off_cfg), ESP_ZB_ZCL_CLUSTER_SERVER_ROLE); + esp_zb_cluster_list_add_identify_cluster(cluster_list, esp_zb_zcl_attr_list_create(ESP_ZB_ZCL_CLUSTER_ID_IDENTIFY), ESP_ZB_ZCL_CLUSTER_SERVER_ROLE); + return cluster_list; +} + +ZigbeeBasicOnOffDevice::ZigbeeBasicOnOffDevice(uint8_t endpoint) : ZigbeeEP(endpoint) { + _device_id = ESP_ZB_HA_SIMPLE_SENSOR_DEVICE_ID; + + //Create custom occupancy sensor configuration + esp_zb_on_off_cfg_t on_off_cfg = ZIGBEE_DEFAULT_ON_OFF_CONFIG(); + _cluster_list = zigbee_on_off_clusters_create(&on_off_cfg); + + _ep_config = {.endpoint = _endpoint, .app_profile_id = ESP_ZB_AF_HA_PROFILE_ID, .app_device_id = ESP_ZB_HA_SIMPLE_SENSOR_DEVICE_ID, .app_device_version = 0}; +} + +//set attribute method -> method overridden in child class +void ZigbeeBasicOnOffDevice::zbAttributeSet(const esp_zb_zcl_set_attr_value_message_t *message) { + //check the data and call right method + if (message->info.cluster == ESP_ZB_ZCL_CLUSTER_ID_ON_OFF) { + if (message->attribute.id == ESP_ZB_ZCL_ATTR_ON_OFF_ON_OFF_ID && message->attribute.data.type == ESP_ZB_ZCL_ATTR_TYPE_BOOL) { + _current_state = *(bool *)message->attribute.data.value; + stateChanged(); + } else { + log_w("Received message ignored. Attribute ID: %d not supported for On/Off Light", message->attribute.id); + } + } else { + log_w("Received message ignored. Cluster ID: %d not supported for On/Off Light", message->info.cluster); + } +} + +void ZigbeeBasicOnOffDevice::stateChanged() { + if (_on_state_change) { + _on_state_change(_current_state); + } else { + log_w("No callback function set for state"); + } +} + +void ZigbeeBasicOnOffDevice::setState(bool state) { + _current_state = state; + stateChanged(); + log_v("Updating on off sensor value..."); + /* Update on off sensor value */ + log_d("Setting state to %d", state); + esp_zb_lock_acquire(portMAX_DELAY); + esp_zb_zcl_set_attribute_val( + _endpoint, ESP_ZB_ZCL_CLUSTER_ID_ON_OFF, ESP_ZB_ZCL_CLUSTER_SERVER_ROLE, ESP_ZB_ZCL_ATTR_ON_OFF_ON_OFF_ID, &state, false + ); + esp_zb_lock_release(); +} + +void ZigbeeBasicOnOffDevice::report() { + /* Send report attributes command */ + esp_zb_zcl_report_attr_cmd_t report_attr_cmd; + report_attr_cmd.address_mode = ESP_ZB_APS_ADDR_MODE_DST_ADDR_ENDP_NOT_PRESENT; + report_attr_cmd.attributeID = ESP_ZB_ZCL_ATTR_ON_OFF_ON_OFF_ID; + report_attr_cmd.direction = ESP_ZB_ZCL_CMD_DIRECTION_TO_CLI; + report_attr_cmd.clusterID = ESP_ZB_ZCL_CLUSTER_ID_ON_OFF; + report_attr_cmd.zcl_basic_cmd.src_endpoint = _endpoint; + + esp_zb_lock_acquire(portMAX_DELAY); + esp_zb_zcl_report_attr_cmd_req(&report_attr_cmd); + esp_zb_lock_release(); + log_v("State report sent"); +} + +#endif //SOC_IEEE802154_SUPPORTED && CONFIG_ZB_ENABLED diff --git a/libraries/Zigbee/src/ep/ZigbeeBasicOnOffDevice.h b/libraries/Zigbee/src/ep/ZigbeeBasicOnOffDevice.h new file mode 100644 index 00000000000..3427e79d04e --- /dev/null +++ b/libraries/Zigbee/src/ep/ZigbeeBasicOnOffDevice.h @@ -0,0 +1,62 @@ +/* Class of Zigbee Pressure sensor endpoint inherited from common EP class */ + +#pragma once + +#include "soc/soc_caps.h" +#include "sdkconfig.h" +#if SOC_IEEE802154_SUPPORTED && CONFIG_ZB_ENABLED + +#include "ZigbeeEP.h" +#include "ha/esp_zigbee_ha_standard.h" + +// clang-format off +#define ZIGBEE_DEFAULT_ON_OFF_CONFIG() \ + { \ + .basic_cfg = \ + { \ + .zcl_version = ESP_ZB_ZCL_BASIC_ZCL_VERSION_DEFAULT_VALUE, \ + .power_source = ESP_ZB_ZCL_BASIC_POWER_SOURCE_DEFAULT_VALUE, \ + }, \ + .identify_cfg = \ + { \ + .identify_time = ESP_ZB_ZCL_IDENTIFY_IDENTIFY_TIME_DEFAULT_VALUE, \ + }, \ + .state_cfg = { \ + .on_off = ESP_ZB_ZCL_ON_OFF_ON_OFF_DEFAULT_VALUE, \ + }, \ + } +// clang-format on + +typedef struct esp_zb_on_off_cfg_s { + esp_zb_basic_cluster_cfg_t basic_cfg; + esp_zb_identify_cluster_cfg_t identify_cfg; + esp_zb_on_off_cluster_cfg_t state_cfg; +} esp_zb_on_off_cfg_t; + +class ZigbeeBasicOnOffDevice : public ZigbeeEP { +public: + ZigbeeBasicOnOffDevice(uint8_t endpoint); + ~ZigbeeBasicOnOffDevice(); + + // Use to set a cb function to be called on light change + void onStateChange(void (*callback)(bool)) { + _on_state_change = callback; + } + // Set the occupancy value. True for occupied, false for unoccupied + void setState(bool state); + + // Report the occupancy value + void report(); + + bool getState() { + return _current_state; + } + +private: + void zbAttributeSet(const esp_zb_zcl_set_attr_value_message_t *message); + void (*_on_state_change)(bool); + void stateChanged(); + bool _current_state; +}; + +#endif //SOC_IEEE802154_SUPPORTED && CONFIG_ZB_ENABLED diff --git a/libraries/Zigbee/src/ep/ZigbeeBasicOnOffSensor.cpp b/libraries/Zigbee/src/ep/ZigbeeBasicOnOffSensor.cpp index 9726c254441..665324c8399 100644 --- a/libraries/Zigbee/src/ep/ZigbeeBasicOnOffSensor.cpp +++ b/libraries/Zigbee/src/ep/ZigbeeBasicOnOffSensor.cpp @@ -1,14 +1,14 @@ #include "ZigbeeBasicOnOffSensor.h" #if SOC_IEEE802154_SUPPORTED && CONFIG_ZB_ENABLED -esp_zb_cluster_list_t *zigbee_on_off_clusters_create(esp_zb_on_off_cfg_t *on_off_sensor) { +esp_zb_cluster_list_t *zigbee_on_off_clusters_create(esp_zb_on_off_sensor_cfg_t *on_off_sensor) { esp_zb_basic_cluster_cfg_t *basic_cfg = on_off_sensor ? &(on_off_sensor->basic_cfg) : NULL; esp_zb_identify_cluster_cfg_t *identify_cfg = on_off_sensor ? &(on_off_sensor->identify_cfg) : NULL; esp_zb_on_off_cluster_cfg_t *on_off_cfg = on_off_sensor ? &(on_off_sensor->state_cfg) : NULL; esp_zb_cluster_list_t *cluster_list = esp_zb_zcl_cluster_list_create(); esp_zb_cluster_list_add_basic_cluster(cluster_list, esp_zb_basic_cluster_create(basic_cfg), ESP_ZB_ZCL_CLUSTER_SERVER_ROLE); esp_zb_cluster_list_add_identify_cluster(cluster_list, esp_zb_identify_cluster_create(identify_cfg), ESP_ZB_ZCL_CLUSTER_SERVER_ROLE); - esp_zb_cluster_list_add_on_off_cluster(cluster_list, esp_zb_on_off_cluster_create(on_off_cfg), ESP_ZB_ZCL_CLUSTER_SERVER_ROLE); + esp_zb_cluster_list_add_on_off_cluster(cluster_list, esp_zb_on_off_cluster_create(on_off_cfg), ESP_ZB_ZCL_CLUSTER_CLIENT_ROLE); esp_zb_cluster_list_add_identify_cluster(cluster_list, esp_zb_zcl_attr_list_create(ESP_ZB_ZCL_CLUSTER_ID_IDENTIFY), ESP_ZB_ZCL_CLUSTER_SERVER_ROLE); return cluster_list; } @@ -17,7 +17,7 @@ ZigbeeBasicOnOffSensor::ZigbeeBasicOnOffSensor(uint8_t endpoint) : ZigbeeEP(endp _device_id = ESP_ZB_HA_SIMPLE_SENSOR_DEVICE_ID; //Create custom occupancy sensor configuration - esp_zb_on_off_cfg_t on_off_cfg = ZIGBEE_DEFAULT_ON_OFF_CONFIG(); + esp_zb_on_off_sensor_cfg_t on_off_cfg = ZIGBEE_DEFAULT_ON_OFF_CONFIG(); _cluster_list = zigbee_on_off_clusters_create(&on_off_cfg); _ep_config = {.endpoint = _endpoint, .app_profile_id = ESP_ZB_AF_HA_PROFILE_ID, .app_device_id = ESP_ZB_HA_SIMPLE_SENSOR_DEVICE_ID, .app_device_version = 0}; @@ -29,7 +29,7 @@ void ZigbeeBasicOnOffSensor::setState(bool state) { log_d("Setting state to %d", state); esp_zb_lock_acquire(portMAX_DELAY); esp_zb_zcl_set_attribute_val( - _endpoint, ESP_ZB_ZCL_CLUSTER_ID_ON_OFF, ESP_ZB_ZCL_CLUSTER_SERVER_ROLE, ESP_ZB_ZCL_ATTR_ON_OFF_ON_OFF_ID, &state, false + _endpoint, ESP_ZB_ZCL_CLUSTER_ID_ON_OFF, ESP_ZB_ZCL_CLUSTER_CLIENT_ROLE, ESP_ZB_ZCL_ATTR_ON_OFF_ON_OFF_ID, &state, false ); esp_zb_lock_release(); } @@ -39,7 +39,7 @@ void ZigbeeBasicOnOffSensor::report() { esp_zb_zcl_report_attr_cmd_t report_attr_cmd; report_attr_cmd.address_mode = ESP_ZB_APS_ADDR_MODE_DST_ADDR_ENDP_NOT_PRESENT; report_attr_cmd.attributeID = ESP_ZB_ZCL_ATTR_ON_OFF_ON_OFF_ID; - report_attr_cmd.direction = ESP_ZB_ZCL_CMD_DIRECTION_TO_CLI; + report_attr_cmd.direction = ESP_ZB_ZCL_CMD_DIRECTION_TO_SRV; report_attr_cmd.clusterID = ESP_ZB_ZCL_CLUSTER_ID_ON_OFF; report_attr_cmd.zcl_basic_cmd.src_endpoint = _endpoint; diff --git a/libraries/Zigbee/src/ep/ZigbeeBasicOnOffSensor.h b/libraries/Zigbee/src/ep/ZigbeeBasicOnOffSensor.h index 38ccfc128e1..8fb75fdc3bc 100644 --- a/libraries/Zigbee/src/ep/ZigbeeBasicOnOffSensor.h +++ b/libraries/Zigbee/src/ep/ZigbeeBasicOnOffSensor.h @@ -23,15 +23,15 @@ }, \ .state_cfg = { \ .on_off = ESP_ZB_ZCL_ON_OFF_ON_OFF_DEFAULT_VALUE, \ - } \ - } \ + }, \ + } // clang-format on -typedef struct esp_zb_on_off_cfg_s { +typedef struct esp_zb_on_off_sensor_cfg_s { esp_zb_basic_cluster_cfg_t basic_cfg; esp_zb_identify_cluster_cfg_t identify_cfg; esp_zb_on_off_cluster_cfg_t state_cfg; -} esp_zb_on_off_cfg_t; +} esp_zb_on_off_sensor_cfg_t; class ZigbeeBasicOnOffSensor : public ZigbeeEP { public: @@ -43,6 +43,7 @@ class ZigbeeBasicOnOffSensor : public ZigbeeEP { // Report the occupancy value void report(); + }; #endif //SOC_IEEE802154_SUPPORTED && CONFIG_ZB_ENABLED diff --git a/libraries/Zigbee/src/ep/ZigbeeLightSensor.h b/libraries/Zigbee/src/ep/ZigbeeLightSensor.h index fa79fd2e305..756df0b78e3 100644 --- a/libraries/Zigbee/src/ep/ZigbeeLightSensor.h +++ b/libraries/Zigbee/src/ep/ZigbeeLightSensor.h @@ -26,7 +26,7 @@ .min_value = ESP_ZB_ZCL_ILLUMINANCE_MEASUREMENT_LIGHT_SENSOR_TYPE_DEFAULT_VALUE, \ .max_value = ESP_ZB_ZCL_ILLUMINANCE_MEASUREMENT_LIGHT_SENSOR_TYPE_DEFAULT_VALUE, \ }, \ - } / + } class ZigbeeLightSensor : public ZigbeeEP { public: