From 88cda9ff59d3ca4a8a227e7b6039597ccfa58d34 Mon Sep 17 00:00:00 2001 From: me-no-dev Date: Tue, 21 Jan 2025 17:14:16 +0200 Subject: [PATCH 001/173] feat(idf): Add initial support for IDF v5.5 and ESP32-C5 --- CMakeLists.txt | 2 +- boards.txt | 212 ++++++++++++++++++++++++++++++ cores/esp32/Esp.cpp | 3 + cores/esp32/HardwareSerial.h | 8 ++ cores/esp32/esp32-hal-cpu.c | 8 +- cores/esp32/esp32-hal-i2c-slave.c | 7 +- cores/esp32/esp32-hal-matrix.c | 2 + cores/esp32/esp32-hal-misc.c | 4 +- cores/esp32/esp32-hal-spi.c | 83 ++++++------ cores/esp32/esp32-hal-uart.c | 22 +++- idf_component.yml | 6 +- libraries/SPI/src/SPI.cpp | 2 +- platform.txt | 9 ++ variants/esp32c5/pins_arduino.h | 42 ++++++ 14 files changed, 357 insertions(+), 53 deletions(-) create mode 100644 variants/esp32c5/pins_arduino.h diff --git a/CMakeLists.txt b/CMakeLists.txt index d2a9162a9ba..dd9544f13e7 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -6,7 +6,7 @@ # idf.py build set(min_supported_idf_version "5.3.0") -set(max_supported_idf_version "5.4.99") +set(max_supported_idf_version "5.5.99") set(idf_version "${IDF_VERSION_MAJOR}.${IDF_VERSION_MINOR}.${IDF_VERSION_PATCH}") if ("${idf_version}" AND NOT "$ENV{ARDUINO_SKIP_IDF_VERSION_CHECK}") diff --git a/boards.txt b/boards.txt index 2afe45f50a6..ea520e60fde 100644 --- a/boards.txt +++ b/boards.txt @@ -161,6 +161,218 @@ esp32c2.menu.EraseFlash.all.upload.erase_cmd=-e ############################################################## +esp32c5.name=ESP32C5 Dev Module + +esp32c5.bootloader.tool=esptool_py +esp32c5.bootloader.tool.default=esptool_py + +esp32c5.upload.tool=esptool_py +esp32c5.upload.tool.default=esptool_py +esp32c5.upload.tool.network=esp_ota + +esp32c5.upload.maximum_size=1310720 +esp32c5.upload.maximum_data_size=327680 +esp32c5.upload.flags= +esp32c5.upload.extra_flags= +esp32c5.upload.use_1200bps_touch=false +esp32c5.upload.wait_for_upload_port=false + +esp32c5.serial.disableDTR=false +esp32c5.serial.disableRTS=false + +esp32c5.build.tarch=riscv32 +esp32c5.build.target=esp +esp32c5.build.mcu=esp32c5 +esp32c5.build.core=esp32 +esp32c5.build.variant=esp32c5 +esp32c5.build.board=ESP32C5_DEV +esp32c5.build.bootloader_addr=0x0 + +esp32c5.build.cdc_on_boot=0 +esp32c5.build.f_cpu=240000000L +esp32c5.build.flash_size=4MB +esp32c5.build.flash_freq=80m +esp32c5.build.flash_mode=qio +esp32c5.build.boot=qio +esp32c5.build.partitions=default +esp32c5.build.defines= + +## IDE 2.0 Seems to not update the value +esp32c5.menu.JTAGAdapter.default=Disabled +esp32c5.menu.JTAGAdapter.default.build.copy_jtag_files=0 +esp32c5.menu.JTAGAdapter.builtin=Integrated USB JTAG +esp32c5.menu.JTAGAdapter.builtin.build.openocdscript=esp32c5-builtin.cfg +esp32c5.menu.JTAGAdapter.builtin.build.copy_jtag_files=1 +esp32c5.menu.JTAGAdapter.external=FTDI Adapter +esp32c5.menu.JTAGAdapter.external.build.openocdscript=esp32c5-ftdi.cfg +esp32c5.menu.JTAGAdapter.external.build.copy_jtag_files=1 +esp32c5.menu.JTAGAdapter.bridge=ESP USB Bridge +esp32c5.menu.JTAGAdapter.bridge.build.openocdscript=esp32c5-bridge.cfg +esp32c5.menu.JTAGAdapter.bridge.build.copy_jtag_files=1 + +esp32c5.menu.CDCOnBoot.default=Disabled +esp32c5.menu.CDCOnBoot.default.build.cdc_on_boot=0 +esp32c5.menu.CDCOnBoot.cdc=Enabled +esp32c5.menu.CDCOnBoot.cdc.build.cdc_on_boot=1 + +esp32c5.menu.PartitionScheme.default=Default 4MB with spiffs (1.2MB APP/1.5MB SPIFFS) +esp32c5.menu.PartitionScheme.default.build.partitions=default +esp32c5.menu.PartitionScheme.defaultffat=Default 4MB with ffat (1.2MB APP/1.5MB FATFS) +esp32c5.menu.PartitionScheme.defaultffat.build.partitions=default_ffat +esp32c5.menu.PartitionScheme.default_8MB=8M with spiffs (3MB APP/1.5MB SPIFFS) +esp32c5.menu.PartitionScheme.default_8MB.build.partitions=default_8MB +esp32c5.menu.PartitionScheme.default_8MB.upload.maximum_size=3342336 +esp32c5.menu.PartitionScheme.minimal=Minimal (1.3MB APP/700KB SPIFFS) +esp32c5.menu.PartitionScheme.minimal.build.partitions=minimal +esp32c5.menu.PartitionScheme.no_fs=No FS 4MB (2MB APP x2) +esp32c5.menu.PartitionScheme.no_fs.build.partitions=no_fs +esp32c5.menu.PartitionScheme.no_fs.upload.maximum_size=2031616 +esp32c5.menu.PartitionScheme.no_ota=No OTA (2MB APP/2MB SPIFFS) +esp32c5.menu.PartitionScheme.no_ota.build.partitions=no_ota +esp32c5.menu.PartitionScheme.no_ota.upload.maximum_size=2097152 +esp32c5.menu.PartitionScheme.noota_3g=No OTA (1MB APP/3MB SPIFFS) +esp32c5.menu.PartitionScheme.noota_3g.build.partitions=noota_3g +esp32c5.menu.PartitionScheme.noota_3g.upload.maximum_size=1048576 +esp32c5.menu.PartitionScheme.noota_ffat=No OTA (2MB APP/2MB FATFS) +esp32c5.menu.PartitionScheme.noota_ffat.build.partitions=noota_ffat +esp32c5.menu.PartitionScheme.noota_ffat.upload.maximum_size=2097152 +esp32c5.menu.PartitionScheme.noota_3gffat=No OTA (1MB APP/3MB FATFS) +esp32c5.menu.PartitionScheme.noota_3gffat.build.partitions=noota_3gffat +esp32c5.menu.PartitionScheme.noota_3gffat.upload.maximum_size=1048576 +esp32c5.menu.PartitionScheme.huge_app=Huge APP (3MB No OTA/1MB SPIFFS) +esp32c5.menu.PartitionScheme.huge_app.build.partitions=huge_app +esp32c5.menu.PartitionScheme.huge_app.upload.maximum_size=3145728 +esp32c5.menu.PartitionScheme.min_spiffs=Minimal SPIFFS (1.9MB APP with OTA/190KB SPIFFS) +esp32c5.menu.PartitionScheme.min_spiffs.build.partitions=min_spiffs +esp32c5.menu.PartitionScheme.min_spiffs.upload.maximum_size=1966080 +esp32c5.menu.PartitionScheme.fatflash=16M Flash (2MB APP/12.5MB FATFS) +esp32c5.menu.PartitionScheme.fatflash.build.partitions=ffat +esp32c5.menu.PartitionScheme.fatflash.upload.maximum_size=2097152 +esp32c5.menu.PartitionScheme.app3M_fat9M_16MB=16M Flash (3MB APP/9.9MB FATFS) +esp32c5.menu.PartitionScheme.app3M_fat9M_16MB.build.partitions=app3M_fat9M_16MB +esp32c5.menu.PartitionScheme.app3M_fat9M_16MB.upload.maximum_size=3145728 +esp32c5.menu.PartitionScheme.rainmaker=RainMaker 4MB +esp32c5.menu.PartitionScheme.rainmaker.build.partitions=rainmaker +esp32c5.menu.PartitionScheme.rainmaker.upload.maximum_size=1966080 +esp32c5.menu.PartitionScheme.rainmaker_4MB=RainMaker 4MB No OTA +esp32c5.menu.PartitionScheme.rainmaker_4MB.build.partitions=rainmaker_4MB_no_ota +esp32c5.menu.PartitionScheme.rainmaker_4MB.upload.maximum_size=4038656 +esp32c5.menu.PartitionScheme.rainmaker_8MB=RainMaker 8MB +esp32c5.menu.PartitionScheme.rainmaker_8MB.build.partitions=rainmaker_8MB +esp32c5.menu.PartitionScheme.rainmaker_8MB.upload.maximum_size=4116480 +esp32c5.menu.PartitionScheme.zigbee_2MB=Zigbee 2MB with spiffs +esp32c5.menu.PartitionScheme.zigbee_2MB.build.partitions=zigbee_2MB +esp32c5.menu.PartitionScheme.zigbee_2MB.upload.maximum_size=1310720 +esp32c5.menu.PartitionScheme.zigbee=Zigbee 4MB with spiffs +esp32c5.menu.PartitionScheme.zigbee.build.partitions=zigbee +esp32c5.menu.PartitionScheme.zigbee.upload.maximum_size=1310720 +esp32c5.menu.PartitionScheme.zigbee_8MB=Zigbee 8MB with spiffs +esp32c5.menu.PartitionScheme.zigbee_8MB.build.partitions=zigbee_8MB +esp32c5.menu.PartitionScheme.zigbee_8MB.upload.maximum_size=3407872 +esp32c5.menu.PartitionScheme.zigbee_zczr_2MB=Zigbee ZCZR 2MB with spiffs +esp32c5.menu.PartitionScheme.zigbee_zczr_2MB.build.partitions=zigbee_zczr_2MB +esp32c5.menu.PartitionScheme.zigbee_zczr_2MB.upload.maximum_size=1310720 +esp32c5.menu.PartitionScheme.zigbee_zczr=Zigbee ZCZR 4MB with spiffs +esp32c5.menu.PartitionScheme.zigbee_zczr.build.partitions=zigbee_zczr +esp32c5.menu.PartitionScheme.zigbee_zczr.upload.maximum_size=1310720 +esp32c5.menu.PartitionScheme.zigbee_zczr_8MB=Zigbee ZCZR 8MB with spiffs +esp32c5.menu.PartitionScheme.zigbee_zczr_8MB.build.partitions=zigbee_zczr_8MB +esp32c5.menu.PartitionScheme.zigbee_zczr_8MB.upload.maximum_size=3407872 +esp32c5.menu.PartitionScheme.custom=Custom +esp32c5.menu.PartitionScheme.custom.build.partitions= +esp32c5.menu.PartitionScheme.custom.upload.maximum_size=16777216 + +esp32c5.menu.CPUFreq.240=240MHz (WiFi) +esp32c5.menu.CPUFreq.240.build.f_cpu=240000000L +esp32c5.menu.CPUFreq.120=120MHz (WiFi) +esp32c5.menu.CPUFreq.120.build.f_cpu=120000000L +esp32c5.menu.CPUFreq.80=80MHz (WiFi) +esp32c5.menu.CPUFreq.80.build.f_cpu=80000000L +esp32c5.menu.CPUFreq.40=40MHz +esp32c5.menu.CPUFreq.40.build.f_cpu=40000000L +esp32c5.menu.CPUFreq.20=20MHz +esp32c5.menu.CPUFreq.20.build.f_cpu=20000000L +esp32c5.menu.CPUFreq.10=10MHz +esp32c5.menu.CPUFreq.10.build.f_cpu=10000000L + +esp32c5.menu.FlashMode.qio=QIO +esp32c5.menu.FlashMode.qio.build.flash_mode=dio +esp32c5.menu.FlashMode.qio.build.boot=qio +esp32c5.menu.FlashMode.dio=DIO +esp32c5.menu.FlashMode.dio.build.flash_mode=dio +esp32c5.menu.FlashMode.dio.build.boot=dio + +esp32c5.menu.FlashFreq.80=80MHz +esp32c5.menu.FlashFreq.80.build.flash_freq=80m +esp32c5.menu.FlashFreq.40=40MHz +esp32c5.menu.FlashFreq.40.build.flash_freq=40m + +esp32c5.menu.FlashSize.4M=4MB (32Mb) +esp32c5.menu.FlashSize.4M.build.flash_size=4MB +esp32c5.menu.FlashSize.8M=8MB (64Mb) +esp32c5.menu.FlashSize.8M.build.flash_size=8MB +esp32c5.menu.FlashSize.2M=2MB (16Mb) +esp32c5.menu.FlashSize.2M.build.flash_size=2MB +esp32c5.menu.FlashSize.16M=16MB (128Mb) +esp32c5.menu.FlashSize.16M.build.flash_size=16MB + +esp32c5.menu.UploadSpeed.921600=921600 +esp32c5.menu.UploadSpeed.921600.upload.speed=921600 +esp32c5.menu.UploadSpeed.115200=115200 +esp32c5.menu.UploadSpeed.115200.upload.speed=115200 +esp32c5.menu.UploadSpeed.256000.windows=256000 +esp32c5.menu.UploadSpeed.256000.upload.speed=256000 +esp32c5.menu.UploadSpeed.230400.windows.upload.speed=256000 +esp32c5.menu.UploadSpeed.230400=230400 +esp32c5.menu.UploadSpeed.230400.upload.speed=230400 +esp32c5.menu.UploadSpeed.460800.linux=460800 +esp32c5.menu.UploadSpeed.460800.macosx=460800 +esp32c5.menu.UploadSpeed.460800.upload.speed=460800 +esp32c5.menu.UploadSpeed.512000.windows=512000 +esp32c5.menu.UploadSpeed.512000.upload.speed=512000 + +esp32c5.menu.DebugLevel.none=None +esp32c5.menu.DebugLevel.none.build.code_debug=0 +esp32c5.menu.DebugLevel.error=Error +esp32c5.menu.DebugLevel.error.build.code_debug=1 +esp32c5.menu.DebugLevel.warn=Warn +esp32c5.menu.DebugLevel.warn.build.code_debug=2 +esp32c5.menu.DebugLevel.info=Info +esp32c5.menu.DebugLevel.info.build.code_debug=3 +esp32c5.menu.DebugLevel.debug=Debug +esp32c5.menu.DebugLevel.debug.build.code_debug=4 +esp32c5.menu.DebugLevel.verbose=Verbose +esp32c5.menu.DebugLevel.verbose.build.code_debug=5 + +esp32c5.menu.EraseFlash.none=Disabled +esp32c5.menu.EraseFlash.none.upload.erase_cmd= +esp32c5.menu.EraseFlash.all=Enabled +esp32c5.menu.EraseFlash.all.upload.erase_cmd=-e + +esp32c5.menu.ZigbeeMode.default=Disabled +esp32c5.menu.ZigbeeMode.default.build.zigbee_mode= +esp32c5.menu.ZigbeeMode.default.build.zigbee_libs= +esp32c5.menu.ZigbeeMode.ed=Zigbee ED (end device) +esp32c5.menu.ZigbeeMode.ed.build.zigbee_mode=-DZIGBEE_MODE_ED +esp32c5.menu.ZigbeeMode.ed.build.zigbee_libs=-lesp_zb_api_ed -lesp_zb_cli_command -lzboss_stack.ed -lzboss_port +esp32c5.menu.ZigbeeMode.zczr=Zigbee ZCZR (coordinator/router) +esp32c5.menu.ZigbeeMode.zczr.build.zigbee_mode=-DZIGBEE_MODE_ZCZR +esp32c5.menu.ZigbeeMode.zczr.build.zigbee_libs=-lesp_zb_api_zczr -lesp_zb_cli_command -lzboss_stack.zczr -lzboss_port +esp32c5.menu.ZigbeeMode.rcp=Zigbee RCP (radio co-processor) +esp32c5.menu.ZigbeeMode.rcp.build.zigbee_mode=-DZIGBEE_MODE_RCP +esp32c5.menu.ZigbeeMode.rcp.build.zigbee_libs=-lesp_zb_api_rcp -lesp_zb_cli_command -lzboss_stack.rcp -lzboss_port +esp32c5.menu.ZigbeeMode.ed_debug=Zigbee ED (end device) - Debug +esp32c5.menu.ZigbeeMode.ed_debug.build.zigbee_mode=-DZIGBEE_MODE_ED +esp32c5.menu.ZigbeeMode.ed_debug.build.zigbee_libs=-lesp_zb_api_ed.debug -lesp_zb_cli_command -lzboss_stack.ed.debug -lzboss_port.debug +esp32c5.menu.ZigbeeMode.zczr_debug=Zigbee ZCZR (coordinator/router) - Debug +esp32c5.menu.ZigbeeMode.zczr_debug.build.zigbee_mode=-DZIGBEE_MODE_ZCZR +esp32c5.menu.ZigbeeMode.zczr_debug.build.zigbee_libs=-lesp_zb_api_zczr.debug -lesp_zb_cli_command -lzboss_stack.zczr.debug -lzboss_port.debug +esp32c5.menu.ZigbeeMode.rcp_debug=Zigbee RCP (radio co-processor) - Debug +esp32c5.menu.ZigbeeMode.rcp_debug.build.zigbee_mode=-DZIGBEE_MODE_RCP +esp32c5.menu.ZigbeeMode.rcp_debug.build.zigbee_libs=-lesp_zb_api_rcp.debug -lesp_zb_cli_command -lzboss_stack.rcp.debug -lzboss_port.debug + +############################################################## + esp32p4.name=ESP32P4 Dev Module esp32p4.bootloader.tool=esptool_py diff --git a/cores/esp32/Esp.cpp b/cores/esp32/Esp.cpp index 9f90a828b25..3cb0e7351c6 100644 --- a/cores/esp32/Esp.cpp +++ b/cores/esp32/Esp.cpp @@ -63,6 +63,9 @@ extern "C" { #elif CONFIG_IDF_TARGET_ESP32P4 #include "esp32p4/rom/spi_flash.h" #define ESP_FLASH_IMAGE_BASE 0x2000 // Esp32p4 is located at 0x2000 +#elif CONFIG_IDF_TARGET_ESP32C5 +#include "esp32c5/rom/spi_flash.h" +#define ESP_FLASH_IMAGE_BASE 0x0000 // Esp32c5 is located at 0x0000 #else #error Target CONFIG_IDF_TARGET is not supported #endif diff --git a/cores/esp32/HardwareSerial.h b/cores/esp32/HardwareSerial.h index b1f6df17724..775b6ee2bcb 100644 --- a/cores/esp32/HardwareSerial.h +++ b/cores/esp32/HardwareSerial.h @@ -139,6 +139,8 @@ typedef enum { #define SOC_RX0 (gpio_num_t)23 #elif CONFIG_IDF_TARGET_ESP32P4 #define SOC_RX0 (gpio_num_t)38 +#elif CONFIG_IDF_TARGET_ESP32C5 +#define SOC_RX0 (gpio_num_t)12 #endif #endif @@ -157,6 +159,8 @@ typedef enum { #define SOC_TX0 (gpio_num_t)24 #elif CONFIG_IDF_TARGET_ESP32P4 #define SOC_TX0 (gpio_num_t)37 +#elif CONFIG_IDF_TARGET_ESP32C5 +#define SOC_TX0 (gpio_num_t)11 #endif #endif @@ -180,6 +184,8 @@ typedef enum { #define RX1 (gpio_num_t)0 #elif CONFIG_IDF_TARGET_ESP32P4 #define RX1 (gpio_num_t)11 +#elif CONFIG_IDF_TARGET_ESP32C5 +#define RX1 (gpio_num_t)4 #endif #endif @@ -200,6 +206,8 @@ typedef enum { #define TX1 (gpio_num_t)1 #elif CONFIG_IDF_TARGET_ESP32P4 #define TX1 (gpio_num_t)10 +#elif CONFIG_IDF_TARGET_ESP32C5 +#define TX1 (gpio_num_t)5 #endif #endif #endif /* SOC_UART_HP_NUM > 1 */ diff --git a/cores/esp32/esp32-hal-cpu.c b/cores/esp32/esp32-hal-cpu.c index 1ffde860792..8c13a4077fd 100644 --- a/cores/esp32/esp32-hal-cpu.c +++ b/cores/esp32/esp32-hal-cpu.c @@ -19,7 +19,7 @@ #include "esp_attr.h" #include "esp_log.h" #include "soc/rtc.h" -#if !defined(CONFIG_IDF_TARGET_ESP32C2) && !defined(CONFIG_IDF_TARGET_ESP32C6) && !defined(CONFIG_IDF_TARGET_ESP32H2) && !defined(CONFIG_IDF_TARGET_ESP32P4) +#if !defined(CONFIG_IDF_TARGET_ESP32C2) && !defined(CONFIG_IDF_TARGET_ESP32C6) && !defined(CONFIG_IDF_TARGET_ESP32H2) && !defined(CONFIG_IDF_TARGET_ESP32P4) && !defined(CONFIG_IDF_TARGET_ESP32C5) #include "soc/rtc_cntl_reg.h" #include "soc/syscon_reg.h" #endif @@ -48,6 +48,8 @@ #include "esp32h2/rom/rtc.h" #elif CONFIG_IDF_TARGET_ESP32P4 #include "esp32p4/rom/rtc.h" +#elif CONFIG_IDF_TARGET_ESP32C5 +#include "esp32c5/rom/rtc.h" #else #error Target CONFIG_IDF_TARGET is not supported #endif @@ -179,7 +181,7 @@ bool setCpuFrequencyMhz(uint32_t cpu_freq_mhz) { rtc_cpu_freq_config_t conf, cconf; uint32_t capb, apb; //Get XTAL Frequency and calculate min CPU MHz -#if (!defined(CONFIG_IDF_TARGET_ESP32H2) && !defined(CONFIG_IDF_TARGET_ESP32P4)) +#if (!defined(CONFIG_IDF_TARGET_ESP32H2) && !defined(CONFIG_IDF_TARGET_ESP32P4) && !defined(CONFIG_IDF_TARGET_ESP32C5)) rtc_xtal_freq_t xtal = rtc_clk_xtal_freq_get(); #endif #if CONFIG_IDF_TARGET_ESP32 @@ -195,7 +197,7 @@ bool setCpuFrequencyMhz(uint32_t cpu_freq_mhz) { } } #endif -#if (!defined(CONFIG_IDF_TARGET_ESP32H2) && !defined(CONFIG_IDF_TARGET_ESP32P4)) +#if (!defined(CONFIG_IDF_TARGET_ESP32H2) && !defined(CONFIG_IDF_TARGET_ESP32P4) && !defined(CONFIG_IDF_TARGET_ESP32C5)) if (cpu_freq_mhz > xtal && cpu_freq_mhz != 240 && cpu_freq_mhz != 160 && cpu_freq_mhz != 120 && cpu_freq_mhz != 80) { if (xtal >= RTC_XTAL_FREQ_40M) { log_e("Bad frequency: %u MHz! Options are: 240, 160, 120, 80, %u, %u and %u MHz", cpu_freq_mhz, xtal, xtal / 2, xtal / 4); diff --git a/cores/esp32/esp32-hal-i2c-slave.c b/cores/esp32/esp32-hal-i2c-slave.c index 46c3a4d58c2..0e01259da61 100644 --- a/cores/esp32/esp32-hal-i2c-slave.c +++ b/cores/esp32/esp32-hal-i2c-slave.c @@ -43,7 +43,9 @@ #include "soc/i2c_struct.h" #include "soc/periph_defs.h" #include "hal/i2c_ll.h" +#ifndef CONFIG_IDF_TARGET_ESP32C5 #include "hal/clk_gate_ll.h" +#endif #include "esp32-hal-log.h" #include "esp32-hal-i2c-slave.h" #include "esp32-hal-periman.h" @@ -325,7 +327,7 @@ esp_err_t i2cSlaveInit(uint8_t num, int sda, int scl, uint16_t slaveID, uint32_t frequency = 100000L; } frequency = (frequency * 5) / 4; -#if !defined(CONFIG_IDF_TARGET_ESP32P4) +#if !defined(CONFIG_IDF_TARGET_ESP32P4) && !defined(CONFIG_IDF_TARGET_ESP32C5) if (i2c->num == 0) { periph_ll_enable_clk_clear_rst(PERIPH_I2C0_MODULE); #if SOC_HP_I2C_NUM > 1 @@ -556,6 +558,9 @@ static bool i2c_slave_set_frequency(i2c_slave_struct_t *i2c, uint32_t clk_speed) i2c_ll_set_source_clk(i2c->dev, SOC_MOD_CLK_APB); /*!< I2C source clock from APB, 80M*/ } #elif SOC_I2C_SUPPORT_XTAL +#ifndef XTAL_CLK_FREQ +#define XTAL_CLK_FREQ APB_CLK_FREQ +#endif i2c_ll_master_cal_bus_clk(XTAL_CLK_FREQ, clk_speed, &clk_cal); I2C_CLOCK_SRC_ATOMIC() { i2c_ll_set_source_clk(i2c->dev, SOC_MOD_CLK_XTAL); /*!< I2C source clock from XTAL, 40M */ diff --git a/cores/esp32/esp32-hal-matrix.c b/cores/esp32/esp32-hal-matrix.c index 7cddb4e04db..0d81e979f2b 100644 --- a/cores/esp32/esp32-hal-matrix.c +++ b/cores/esp32/esp32-hal-matrix.c @@ -34,6 +34,8 @@ #include "esp32h2/rom/gpio.h" #elif CONFIG_IDF_TARGET_ESP32P4 #include "esp32p4/rom/gpio.h" +#elif CONFIG_IDF_TARGET_ESP32C5 +#include "esp32c5/rom/gpio.h" #else #error Target CONFIG_IDF_TARGET is not supported #endif diff --git a/cores/esp32/esp32-hal-misc.c b/cores/esp32/esp32-hal-misc.c index 50e2973d27a..39f81d50ae5 100644 --- a/cores/esp32/esp32-hal-misc.c +++ b/cores/esp32/esp32-hal-misc.c @@ -30,7 +30,7 @@ #endif //CONFIG_BT_ENABLED #include #include "soc/rtc.h" -#if !defined(CONFIG_IDF_TARGET_ESP32C2) && !defined(CONFIG_IDF_TARGET_ESP32C6) && !defined(CONFIG_IDF_TARGET_ESP32H2) && !defined(CONFIG_IDF_TARGET_ESP32P4) +#if !defined(CONFIG_IDF_TARGET_ESP32C2) && !defined(CONFIG_IDF_TARGET_ESP32C6) && !defined(CONFIG_IDF_TARGET_ESP32H2) && !defined(CONFIG_IDF_TARGET_ESP32P4) && !defined(CONFIG_IDF_TARGET_ESP32C5) #include "soc/rtc_cntl_reg.h" #include "soc/syscon_reg.h" #endif @@ -56,6 +56,8 @@ #include "esp32h2/rom/rtc.h" #elif CONFIG_IDF_TARGET_ESP32P4 #include "esp32p4/rom/rtc.h" +#elif CONFIG_IDF_TARGET_ESP32C5 +#include "esp32c5/rom/rtc.h" #else #error Target CONFIG_IDF_TARGET is not supported diff --git a/cores/esp32/esp32-hal-spi.c b/cores/esp32/esp32-hal-spi.c index 80928309670..4cff353d76c 100644 --- a/cores/esp32/esp32-hal-spi.c +++ b/cores/esp32/esp32-hal-spi.c @@ -26,7 +26,9 @@ #include "soc/io_mux_reg.h" #include "soc/gpio_sig_map.h" #include "soc/rtc.h" +#ifndef CONFIG_IDF_TARGET_ESP32C5 #include "hal/clk_gate_ll.h" +#endif #include "esp32-hal-periman.h" #include "esp_private/periph_ctrl.h" @@ -60,6 +62,9 @@ #elif CONFIG_IDF_TARGET_ESP32P4 #include "esp32p4/rom/ets_sys.h" #include "esp32p4/rom/gpio.h" +#elif CONFIG_IDF_TARGET_ESP32C5 +#include "esp32c5/rom/ets_sys.h" +#include "esp32c5/rom/gpio.h" #else #error Target CONFIG_IDF_TARGET is not supported #endif @@ -119,7 +124,7 @@ struct spi_struct_t { #define SPI_SS_IDX(p, n) ((p == 0) ? SPI_FSPI_SS_IDX(n) : ((p == 1) ? SPI_HSPI_SS_IDX(n) : 0)) -#elif CONFIG_IDF_TARGET_ESP32C2 || CONFIG_IDF_TARGET_ESP32C3 || CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 +#elif CONFIG_IDF_TARGET_ESP32C2 || CONFIG_IDF_TARGET_ESP32C3 || CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32C5 // ESP32C3 #define SPI_COUNT (1) @@ -161,7 +166,7 @@ static spi_t _spi_bus_array[] = { {(volatile spi_dev_t *)(DR_REG_SPI2_BASE), 0, -1, -1, -1, -1} #elif CONFIG_IDF_TARGET_ESP32C3 {(volatile spi_dev_t *)(DR_REG_SPI2_BASE), 0, -1, -1, -1, -1} -#elif CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 +#elif CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32C5 {(spi_dev_t *)(DR_REG_SPI2_BASE), 0, -1, -1, -1, -1} #else {(volatile spi_dev_t *)(DR_REG_SPI0_BASE), 0, -1, -1, -1, -1}, @@ -188,7 +193,7 @@ static spi_t _spi_bus_array[] = { {(volatile spi_dev_t *)(DR_REG_SPI2_BASE), NULL, 0, -1, -1, -1, -1} #elif CONFIG_IDF_TARGET_ESP32C3 {(volatile spi_dev_t *)(DR_REG_SPI2_BASE), NULL, 0, -1, -1, -1, -1} -#elif CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 +#elif CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32C5 {(spi_dev_t *)(DR_REG_SPI2_BASE), NULL, 0, -1, -1, -1, -1} #else {(volatile spi_dev_t *)(DR_REG_SPI0_BASE), NULL, 0, -1, -1, -1, -1}, @@ -685,7 +690,7 @@ spi_t *spiStartBus(uint8_t spi_num, uint32_t clockDiv, uint8_t dataMode, uint8_t spi->dev->user.doutdin = 1; int i; for (i = 0; i < 16; i++) { -#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 +#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 || CONFIG_IDF_TARGET_ESP32C5 spi->dev->data_buf[i].val = 0x00000000; #else spi->dev->data_buf[i] = 0x00000000; @@ -733,7 +738,7 @@ void spiWrite(spi_t *spi, const uint32_t *data, uint8_t len) { spi->dev->miso_dlen.usr_miso_dbitlen = 0; #endif for (i = 0; i < len; i++) { -#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 +#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 || CONFIG_IDF_TARGET_ESP32C5 spi->dev->data_buf[i].val = data[i]; #else spi->dev->data_buf[i] = data[i]; @@ -760,7 +765,7 @@ void spiTransfer(spi_t *spi, uint32_t *data, uint8_t len) { spi->dev->mosi_dlen.usr_mosi_dbitlen = (len * 32) - 1; spi->dev->miso_dlen.usr_miso_dbitlen = (len * 32) - 1; for (i = 0; i < len; i++) { -#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 +#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 || CONFIG_IDF_TARGET_ESP32C5 spi->dev->data_buf[i].val = data[i]; #else spi->dev->data_buf[i] = data[i]; @@ -773,7 +778,7 @@ void spiTransfer(spi_t *spi, uint32_t *data, uint8_t len) { spi->dev->cmd.usr = 1; while (spi->dev->cmd.usr); for (i = 0; i < len; i++) { -#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 +#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 || CONFIG_IDF_TARGET_ESP32C5 data[i] = spi->dev->data_buf[i].val; #else data[i] = spi->dev->data_buf[i]; @@ -791,7 +796,7 @@ void spiWriteByte(spi_t *spi, uint8_t data) { #if CONFIG_IDF_TARGET_ESP32S2 || CONFIG_IDF_TARGET_ESP32 spi->dev->miso_dlen.usr_miso_dbitlen = 0; #endif -#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 +#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 || CONFIG_IDF_TARGET_ESP32C5 spi->dev->data_buf[0].val = data; #else spi->dev->data_buf[0] = data; @@ -813,7 +818,7 @@ uint8_t spiTransferByte(spi_t *spi, uint8_t data) { SPI_MUTEX_LOCK(); spi->dev->mosi_dlen.usr_mosi_dbitlen = 7; spi->dev->miso_dlen.usr_miso_dbitlen = 7; -#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 +#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 || CONFIG_IDF_TARGET_ESP32C5 spi->dev->data_buf[0].val = data; #else spi->dev->data_buf[0] = data; @@ -824,7 +829,7 @@ uint8_t spiTransferByte(spi_t *spi, uint8_t data) { #endif spi->dev->cmd.usr = 1; while (spi->dev->cmd.usr); -#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 +#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 || CONFIG_IDF_TARGET_ESP32C5 data = spi->dev->data_buf[0].val & 0xFF; #else data = spi->dev->data_buf[0] & 0xFF; @@ -854,7 +859,7 @@ void spiWriteWord(spi_t *spi, uint16_t data) { #if CONFIG_IDF_TARGET_ESP32S2 || CONFIG_IDF_TARGET_ESP32 spi->dev->miso_dlen.usr_miso_dbitlen = 0; #endif -#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 +#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 || CONFIG_IDF_TARGET_ESP32C5 spi->dev->data_buf[0].val = data; #else spi->dev->data_buf[0] = data; @@ -878,7 +883,7 @@ uint16_t spiTransferWord(spi_t *spi, uint16_t data) { SPI_MUTEX_LOCK(); spi->dev->mosi_dlen.usr_mosi_dbitlen = 15; spi->dev->miso_dlen.usr_miso_dbitlen = 15; -#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 +#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 || CONFIG_IDF_TARGET_ESP32C5 spi->dev->data_buf[0].val = data; #else spi->dev->data_buf[0] = data; @@ -889,7 +894,7 @@ uint16_t spiTransferWord(spi_t *spi, uint16_t data) { #endif spi->dev->cmd.usr = 1; while (spi->dev->cmd.usr); -#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 +#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 || CONFIG_IDF_TARGET_ESP32C5 data = spi->dev->data_buf[0].val; #else data = spi->dev->data_buf[0]; @@ -913,7 +918,7 @@ void spiWriteLong(spi_t *spi, uint32_t data) { #if CONFIG_IDF_TARGET_ESP32S2 || CONFIG_IDF_TARGET_ESP32 spi->dev->miso_dlen.usr_miso_dbitlen = 0; #endif -#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 +#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 || CONFIG_IDF_TARGET_ESP32C5 spi->dev->data_buf[0].val = data; #else spi->dev->data_buf[0] = data; @@ -937,7 +942,7 @@ uint32_t spiTransferLong(spi_t *spi, uint32_t data) { SPI_MUTEX_LOCK(); spi->dev->mosi_dlen.usr_mosi_dbitlen = 31; spi->dev->miso_dlen.usr_miso_dbitlen = 31; -#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 +#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 || CONFIG_IDF_TARGET_ESP32C5 spi->dev->data_buf[0].val = data; #else spi->dev->data_buf[0] = data; @@ -948,7 +953,7 @@ uint32_t spiTransferLong(spi_t *spi, uint32_t data) { #endif spi->dev->cmd.usr = 1; while (spi->dev->cmd.usr); -#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 +#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 || CONFIG_IDF_TARGET_ESP32C5 data = spi->dev->data_buf[0].val; #else data = spi->dev->data_buf[0]; @@ -987,7 +992,7 @@ static void __spiTransferBytes(spi_t *spi, const uint8_t *data, uint8_t *out, ui spi->dev->miso_dlen.usr_miso_dbitlen = ((bytes * 8) - 1); for (i = 0; i < words; i++) { -#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 +#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 || CONFIG_IDF_TARGET_ESP32C5 spi->dev->data_buf[i].val = wordsBuf[i]; //copy buffer to spi fifo #else spi->dev->data_buf[i] = wordsBuf[i]; //copy buffer to spi fifo @@ -1004,7 +1009,7 @@ static void __spiTransferBytes(spi_t *spi, const uint8_t *data, uint8_t *out, ui if (out) { for (i = 0; i < words; i++) { -#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 +#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 || CONFIG_IDF_TARGET_ESP32C5 wordsBuf[i] = spi->dev->data_buf[i].val; //copy spi fifo to buffer #else wordsBuf[i] = spi->dev->data_buf[i]; //copy spi fifo to buffer @@ -1145,7 +1150,7 @@ void ARDUINO_ISR_ATTR spiWriteByteNL(spi_t *spi, uint8_t data) { #if CONFIG_IDF_TARGET_ESP32S2 || CONFIG_IDF_TARGET_ESP32 spi->dev->miso_dlen.usr_miso_dbitlen = 0; #endif -#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 +#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 || CONFIG_IDF_TARGET_ESP32C5 spi->dev->data_buf[0].val = data; #else spi->dev->data_buf[0] = data; @@ -1164,7 +1169,7 @@ uint8_t spiTransferByteNL(spi_t *spi, uint8_t data) { } spi->dev->mosi_dlen.usr_mosi_dbitlen = 7; spi->dev->miso_dlen.usr_miso_dbitlen = 7; -#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 +#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 || CONFIG_IDF_TARGET_ESP32C5 spi->dev->data_buf[0].val = data; #else spi->dev->data_buf[0] = data; @@ -1175,7 +1180,7 @@ uint8_t spiTransferByteNL(spi_t *spi, uint8_t data) { #endif spi->dev->cmd.usr = 1; while (spi->dev->cmd.usr); -#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 +#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 || CONFIG_IDF_TARGET_ESP32C5 data = spi->dev->data_buf[0].val & 0xFF; #else data = spi->dev->data_buf[0] & 0xFF; @@ -1194,7 +1199,7 @@ void ARDUINO_ISR_ATTR spiWriteShortNL(spi_t *spi, uint16_t data) { #if CONFIG_IDF_TARGET_ESP32S2 || CONFIG_IDF_TARGET_ESP32 spi->dev->miso_dlen.usr_miso_dbitlen = 0; #endif -#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 +#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 || CONFIG_IDF_TARGET_ESP32C5 spi->dev->data_buf[0].val = data; #else spi->dev->data_buf[0] = data; @@ -1216,7 +1221,7 @@ uint16_t spiTransferShortNL(spi_t *spi, uint16_t data) { } spi->dev->mosi_dlen.usr_mosi_dbitlen = 15; spi->dev->miso_dlen.usr_miso_dbitlen = 15; -#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 +#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 || CONFIG_IDF_TARGET_ESP32C5 spi->dev->data_buf[0].val = data; #else spi->dev->data_buf[0] = data; @@ -1227,7 +1232,7 @@ uint16_t spiTransferShortNL(spi_t *spi, uint16_t data) { #endif spi->dev->cmd.usr = 1; while (spi->dev->cmd.usr); -#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 +#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 || CONFIG_IDF_TARGET_ESP32C5 data = spi->dev->data_buf[0].val & 0xFFFF; #else data = spi->dev->data_buf[0] & 0xFFFF; @@ -1249,7 +1254,7 @@ void ARDUINO_ISR_ATTR spiWriteLongNL(spi_t *spi, uint32_t data) { #if CONFIG_IDF_TARGET_ESP32S2 || CONFIG_IDF_TARGET_ESP32 spi->dev->miso_dlen.usr_miso_dbitlen = 0; #endif -#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 +#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 || CONFIG_IDF_TARGET_ESP32C5 spi->dev->data_buf[0].val = data; #else spi->dev->data_buf[0] = data; @@ -1271,7 +1276,7 @@ uint32_t spiTransferLongNL(spi_t *spi, uint32_t data) { } spi->dev->mosi_dlen.usr_mosi_dbitlen = 31; spi->dev->miso_dlen.usr_miso_dbitlen = 31; -#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 +#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 || CONFIG_IDF_TARGET_ESP32C5 spi->dev->data_buf[0].val = data; #else spi->dev->data_buf[0] = data; @@ -1282,7 +1287,7 @@ uint32_t spiTransferLongNL(spi_t *spi, uint32_t data) { #endif spi->dev->cmd.usr = 1; while (spi->dev->cmd.usr); -#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 +#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 || CONFIG_IDF_TARGET_ESP32C5 data = spi->dev->data_buf[0].val; #else data = spi->dev->data_buf[0]; @@ -1313,7 +1318,7 @@ void spiWriteNL(spi_t *spi, const void *data_in, uint32_t len) { spi->dev->miso_dlen.usr_miso_dbitlen = 0; #endif for (size_t i = 0; i < c_longs; i++) { -#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 +#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 || CONFIG_IDF_TARGET_ESP32C5 spi->dev->data_buf[i].val = data[i]; #else spi->dev->data_buf[i] = data[i]; @@ -1352,7 +1357,7 @@ void spiTransferBytesNL(spi_t *spi, const void *data_in, uint8_t *data_out, uint spi->dev->miso_dlen.usr_miso_dbitlen = (c_len * 8) - 1; if (data) { for (size_t i = 0; i < c_longs; i++) { -#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 +#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 || CONFIG_IDF_TARGET_ESP32C5 spi->dev->data_buf[i].val = data[i]; #else spi->dev->data_buf[i] = data[i]; @@ -1360,7 +1365,7 @@ void spiTransferBytesNL(spi_t *spi, const void *data_in, uint8_t *data_out, uint } } else { for (size_t i = 0; i < c_longs; i++) { -#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 +#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 || CONFIG_IDF_TARGET_ESP32C5 spi->dev->data_buf[i].val = 0xFFFFFFFF; #else spi->dev->data_buf[i] = 0xFFFFFFFF; @@ -1376,13 +1381,13 @@ void spiTransferBytesNL(spi_t *spi, const void *data_in, uint8_t *data_out, uint if (result) { if (c_len & 3) { for (size_t i = 0; i < (c_longs - 1); i++) { -#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 +#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 || CONFIG_IDF_TARGET_ESP32C5 result[i] = spi->dev->data_buf[i].val; #else result[i] = spi->dev->data_buf[i]; #endif } -#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 +#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 || CONFIG_IDF_TARGET_ESP32C5 uint32_t last_data = spi->dev->data_buf[c_longs - 1].val; #else uint32_t last_data = spi->dev->data_buf[c_longs - 1]; @@ -1394,7 +1399,7 @@ void spiTransferBytesNL(spi_t *spi, const void *data_in, uint8_t *data_out, uint } } else { for (size_t i = 0; i < c_longs; i++) { -#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 +#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 || CONFIG_IDF_TARGET_ESP32C5 result[i] = spi->dev->data_buf[i].val; #else result[i] = spi->dev->data_buf[i]; @@ -1436,7 +1441,7 @@ void spiTransferBitsNL(spi_t *spi, uint32_t data, uint32_t *out, uint8_t bits) { spi->dev->mosi_dlen.usr_mosi_dbitlen = (bits - 1); spi->dev->miso_dlen.usr_miso_dbitlen = (bits - 1); -#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 +#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 || CONFIG_IDF_TARGET_ESP32C5 spi->dev->data_buf[0].val = data; #else spi->dev->data_buf[0] = data; @@ -1447,7 +1452,7 @@ void spiTransferBitsNL(spi_t *spi, uint32_t data, uint32_t *out, uint8_t bits) { #endif spi->dev->cmd.usr = 1; while (spi->dev->cmd.usr); -#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 +#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 || CONFIG_IDF_TARGET_ESP32C5 data = spi->dev->data_buf[0].val; #else data = spi->dev->data_buf[0]; @@ -1488,27 +1493,27 @@ void ARDUINO_ISR_ATTR spiWritePixelsNL(spi_t *spi, const void *data_in, uint32_t if (msb) { if (l_bytes && i == (c_longs - 1)) { if (l_bytes == 2) { -#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 +#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 || CONFIG_IDF_TARGET_ESP32C5 MSB_16_SET(spi->dev->data_buf[i].val, data[i]); #else MSB_16_SET(spi->dev->data_buf[i], data[i]); #endif } else { -#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 +#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 || CONFIG_IDF_TARGET_ESP32C5 spi->dev->data_buf[i].val = data[i] & 0xFF; #else spi->dev->data_buf[i] = data[i] & 0xFF; #endif } } else { -#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 +#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 || CONFIG_IDF_TARGET_ESP32C5 MSB_PIX_SET(spi->dev->data_buf[i].val, data[i]); #else MSB_PIX_SET(spi->dev->data_buf[i], data[i]); #endif } } else { -#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 +#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 || CONFIG_IDF_TARGET_ESP32C5 spi->dev->data_buf[i].val = data[i]; #else spi->dev->data_buf[i] = data[i]; diff --git a/cores/esp32/esp32-hal-uart.c b/cores/esp32/esp32-hal-uart.c index 75e2da013ea..a928a8ede3a 100644 --- a/cores/esp32/esp32-hal-uart.c +++ b/cores/esp32/esp32-hal-uart.c @@ -206,6 +206,14 @@ static bool lpuartCheckPins(int8_t rxPin, int8_t txPin, int8_t ctsPin, int8_t rt } #endif // SOC_UART_LP_NUM >= 1 +#ifndef GPIO_FUNC_IN_LOW +#define GPIO_FUNC_IN_LOW GPIO_MATRIX_CONST_ZERO_INPUT +#endif + +#ifndef GPIO_FUNC_IN_HIGH +#define GPIO_FUNC_IN_HIGH GPIO_MATRIX_CONST_ONE_INPUT +#endif + // Negative Pin Number will keep it unmodified, thus this function can detach individual pins // This function will also unset the pins in the Peripheral Manager and set the pin to -1 after detaching static bool _uartDetachPins(uint8_t uart_num, int8_t rxPin, int8_t txPin, int8_t ctsPin, int8_t rtsPin) { @@ -221,7 +229,8 @@ static bool _uartDetachPins(uint8_t uart_num, int8_t rxPin, int8_t txPin, int8_t // detaches HP and LP pins and sets Peripheral Manager and UART information if (rxPin >= 0 && uart->_rxPin == rxPin && perimanGetPinBusType(rxPin) == ESP32_BUS_TYPE_UART_RX) { - gpio_hal_iomux_func_sel(GPIO_PIN_MUX_REG[rxPin], PIN_FUNC_GPIO); + //gpio_hal_iomux_func_sel(GPIO_PIN_MUX_REG[rxPin], PIN_FUNC_GPIO); + esp_rom_gpio_pad_select_gpio(rxPin); // avoids causing BREAK in the UART line if (uart->_inverted) { esp_rom_gpio_connect_in_signal(GPIO_FUNC_IN_LOW, UART_PERIPH_SIGNAL(uart_num, SOC_UART_RX_PIN_IDX), false); @@ -235,7 +244,8 @@ static bool _uartDetachPins(uint8_t uart_num, int8_t rxPin, int8_t txPin, int8_t } } if (txPin >= 0 && uart->_txPin == txPin && perimanGetPinBusType(txPin) == ESP32_BUS_TYPE_UART_TX) { - gpio_hal_iomux_func_sel(GPIO_PIN_MUX_REG[txPin], PIN_FUNC_GPIO); + //gpio_hal_iomux_func_sel(GPIO_PIN_MUX_REG[txPin], PIN_FUNC_GPIO); + esp_rom_gpio_pad_select_gpio(txPin); esp_rom_gpio_connect_out_signal(txPin, SIG_GPIO_OUT_IDX, false, false); uart->_txPin = -1; // -1 means unassigned/detached if (!perimanClearPinBus(txPin)) { @@ -244,7 +254,8 @@ static bool _uartDetachPins(uint8_t uart_num, int8_t rxPin, int8_t txPin, int8_t } } if (ctsPin >= 0 && uart->_ctsPin == ctsPin && perimanGetPinBusType(ctsPin) == ESP32_BUS_TYPE_UART_CTS) { - gpio_hal_iomux_func_sel(GPIO_PIN_MUX_REG[ctsPin], PIN_FUNC_GPIO); + //gpio_hal_iomux_func_sel(GPIO_PIN_MUX_REG[ctsPin], PIN_FUNC_GPIO); + esp_rom_gpio_pad_select_gpio(ctsPin); esp_rom_gpio_connect_in_signal(GPIO_FUNC_IN_LOW, UART_PERIPH_SIGNAL(uart_num, SOC_UART_CTS_PIN_IDX), false); uart->_ctsPin = -1; // -1 means unassigned/detached if (!perimanClearPinBus(ctsPin)) { @@ -253,7 +264,8 @@ static bool _uartDetachPins(uint8_t uart_num, int8_t rxPin, int8_t txPin, int8_t } } if (rtsPin >= 0 && uart->_rtsPin == rtsPin && perimanGetPinBusType(rtsPin) == ESP32_BUS_TYPE_UART_RTS) { - gpio_hal_iomux_func_sel(GPIO_PIN_MUX_REG[rtsPin], PIN_FUNC_GPIO); + //gpio_hal_iomux_func_sel(GPIO_PIN_MUX_REG[rtsPin], PIN_FUNC_GPIO); + esp_rom_gpio_pad_select_gpio(rtsPin); esp_rom_gpio_connect_out_signal(rtsPin, SIG_GPIO_OUT_IDX, false, false); uart->_rtsPin = -1; // -1 means unassigned/detached if (!perimanClearPinBus(rtsPin)) { @@ -808,7 +820,7 @@ void uartSetRxInvert(uart_t *uart, bool invert) { if (uart == NULL) { return; } -#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 +#if CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 || CONFIG_IDF_TARGET_ESP32C5 // POTENTIAL ISSUE :: original code only set/reset rxd_inv bit // IDF or LL set/reset the whole inv_mask! // if (invert) diff --git a/idf_component.yml b/idf_component.yml index 967c4ecf0f6..c1fc614cae5 100644 --- a/idf_component.yml +++ b/idf_component.yml @@ -10,6 +10,7 @@ targets: - esp32c6 - esp32h2 - esp32p4 + - esp32c5 tags: - arduino files: @@ -22,6 +23,7 @@ files: - "variants/esp32c6/**/*" - "variants/esp32h2/**/*" - "variants/esp32p4/**/*" + - "variants/esp32c5/**/*" exclude: - "docs/" - "docs/**/*" @@ -44,7 +46,7 @@ files: - "platform.txt" - "programmers.txt" dependencies: - idf: ">=5.3,<5.5" + idf: ">=5.3,<5.6" # mdns 1.2.1 is necessary to build H2 with no WiFi espressif/mdns: version: "^1.2.3" @@ -107,7 +109,7 @@ dependencies: rules: - if: "target == esp32p4" espressif/esp_wifi_remote: - version: "^0.4.1" + version: "^0.5.4" rules: - if: "target == esp32p4" espressif/libsodium: diff --git a/libraries/SPI/src/SPI.cpp b/libraries/SPI/src/SPI.cpp index 35e52f43e4d..732128809c9 100644 --- a/libraries/SPI/src/SPI.cpp +++ b/libraries/SPI/src/SPI.cpp @@ -83,7 +83,7 @@ void SPIClass::begin(int8_t sck, int8_t miso, int8_t mosi, int8_t ss) { _miso = (_spi_num == FSPI) ? MISO : -1; _mosi = (_spi_num == FSPI) ? MOSI : -1; _ss = (_spi_num == FSPI) ? SS : -1; -#elif CONFIG_IDF_TARGET_ESP32C2 || CONFIG_IDF_TARGET_ESP32C3 || CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 +#elif CONFIG_IDF_TARGET_ESP32C2 || CONFIG_IDF_TARGET_ESP32C3 || CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 || CONFIG_IDF_TARGET_ESP32C5 _sck = SCK; _miso = MISO; _mosi = MOSI; diff --git a/platform.txt b/platform.txt index 65be05b3bf4..f41a8b4a764 100644 --- a/platform.txt +++ b/platform.txt @@ -84,6 +84,7 @@ build.extra_flags.esp32c3=-DARDUINO_USB_MODE=1 -DARDUINO_USB_CDC_ON_BOOT={build. build.extra_flags.esp32c6=-DARDUINO_USB_MODE=1 -DARDUINO_USB_CDC_ON_BOOT={build.cdc_on_boot} build.extra_flags.esp32h2=-DARDUINO_USB_MODE=1 -DARDUINO_USB_CDC_ON_BOOT={build.cdc_on_boot} build.extra_flags.esp32p4=-DARDUINO_USB_MODE={build.usb_mode} -DARDUINO_USB_CDC_ON_BOOT={build.cdc_on_boot} -DARDUINO_USB_MSC_ON_BOOT={build.msc_on_boot} -DARDUINO_USB_DFU_ON_BOOT={build.dfu_on_boot} +build.extra_flags.esp32c5=-DARDUINO_USB_MODE=1 -DARDUINO_USB_CDC_ON_BOOT={build.cdc_on_boot} # This can be overriden in boards.txt build.zigbee_mode= @@ -260,6 +261,14 @@ debug_config.esp32c6= debug_script.esp32h2=esp32h2-builtin.cfg debug_config.esp32h2= +# ESP32-P4 debug configuration (TBD) +debug_script.esp32p4=esp32p4-builtin.cfg +debug_config.esp32p4= + +# ESP32-C5 debug configuration (TBD) +debug_script.esp32c5=esp32c5-builtin.cfg +debug_config.esp32c5= + # Debug API variable definitions debug.executable={build.path}/{build.project_name}.elf debug.toolchain=gcc diff --git a/variants/esp32c5/pins_arduino.h b/variants/esp32c5/pins_arduino.h new file mode 100644 index 00000000000..eeaf3e01ebb --- /dev/null +++ b/variants/esp32c5/pins_arduino.h @@ -0,0 +1,42 @@ +#ifndef Pins_Arduino_h +#define Pins_Arduino_h + +#include +#include "soc/soc_caps.h" + +#define PIN_RGB_LED 8 +// BUILTIN_LED can be used in new Arduino API digitalWrite() like in Blink.ino +static const uint8_t LED_BUILTIN = SOC_GPIO_PIN_COUNT + PIN_RGB_LED; +#define BUILTIN_LED LED_BUILTIN // backward compatibility +#define LED_BUILTIN LED_BUILTIN // allow testing #ifdef LED_BUILTIN +// RGB_BUILTIN and RGB_BRIGHTNESS can be used in new Arduino API rgbLedWrite() +#define RGB_BUILTIN LED_BUILTIN +#define RGB_BRIGHTNESS 64 + +static const uint8_t TX = 11; +static const uint8_t RX = 12; + +// static const uint8_t USB_DM = 13; +// static const uint8_t USB_DP = 14; + +static const uint8_t SDA = 23; +static const uint8_t SCL = 22; + +static const uint8_t SS = 18; +static const uint8_t MOSI = 19; +static const uint8_t MISO = 20; +static const uint8_t SCK = 21; + +static const uint8_t A0 = 1; +static const uint8_t A1 = 2; +static const uint8_t A2 = 3; +static const uint8_t A3 = 4; +static const uint8_t A4 = 5; +static const uint8_t A5 = 6; + +// LP I2C Pins are fixed on ESP32-C6 +#define WIRE1_PIN_DEFINED +static const uint8_t SDA1 = 2; +static const uint8_t SCL1 = 3; + +#endif /* Pins_Arduino_h */ From bf90cbd18316603a717fa08e2ba4dfcb85541236 Mon Sep 17 00:00:00 2001 From: me-no-dev Date: Wed, 22 Jan 2025 01:34:44 +0200 Subject: [PATCH 002/173] fix(examples): Add changes required to some examples --- libraries/ESP32/examples/DeepSleep/TouchWakeUp/TouchWakeUp.ino | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/libraries/ESP32/examples/DeepSleep/TouchWakeUp/TouchWakeUp.ino b/libraries/ESP32/examples/DeepSleep/TouchWakeUp/TouchWakeUp.ino index 9d2b248ba44..5b1e0e9e115 100644 --- a/libraries/ESP32/examples/DeepSleep/TouchWakeUp/TouchWakeUp.ino +++ b/libraries/ESP32/examples/DeepSleep/TouchWakeUp/TouchWakeUp.ino @@ -48,7 +48,7 @@ Method to print the touchpad by which ESP32 has been awaken from sleep */ void print_wakeup_touchpad() { - touchPin = esp_sleep_get_touchpad_wakeup_status(); + touchPin = (touch_pad_t)esp_sleep_get_touchpad_wakeup_status(); #if CONFIG_IDF_TARGET_ESP32 switch (touchPin) { From af5abd5f6196ab4868b229f1e45f662961e531c7 Mon Sep 17 00:00:00 2001 From: me-no-dev Date: Wed, 22 Jan 2025 12:19:42 +0200 Subject: [PATCH 003/173] fix(c5): Update bootloader location --- boards.txt | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/boards.txt b/boards.txt index ea520e60fde..af3e217df9d 100644 --- a/boards.txt +++ b/boards.txt @@ -186,7 +186,7 @@ esp32c5.build.mcu=esp32c5 esp32c5.build.core=esp32 esp32c5.build.variant=esp32c5 esp32c5.build.board=ESP32C5_DEV -esp32c5.build.bootloader_addr=0x0 +esp32c5.build.bootloader_addr=0x2000 esp32c5.build.cdc_on_boot=0 esp32c5.build.f_cpu=240000000L From 0894d7db68562d4690c5b201df7f0c333c776bc1 Mon Sep 17 00:00:00 2001 From: me-no-dev Date: Wed, 22 Jan 2025 12:31:54 +0200 Subject: [PATCH 004/173] fix(report): Add missing chip names --- cores/esp32/Esp.cpp | 6 ++++++ cores/esp32/chip-debug-report.cpp | 5 +++++ 2 files changed, 11 insertions(+) diff --git a/cores/esp32/Esp.cpp b/cores/esp32/Esp.cpp index 3cb0e7351c6..2b2e1280683 100644 --- a/cores/esp32/Esp.cpp +++ b/cores/esp32/Esp.cpp @@ -21,6 +21,7 @@ #include "Esp.h" #include "esp_sleep.h" #include "spi_flash_mmap.h" +#include "esp_idf_version.h" #include #include #include @@ -304,6 +305,11 @@ const char *EspClass::getChipModel(void) { case CHIP_ESP32C6: return "ESP32-C6"; case CHIP_ESP32H2: return "ESP32-H2"; case CHIP_ESP32P4: return "ESP32-P4"; +#if ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 5, 0) + case CHIP_ESP32C5: return "ESP32-C5"; + case CHIP_ESP32C61: return "ESP32-C61"; + case CHIP_ESP32H21: return "ESP32-H21"; +#endif default: return "UNKNOWN"; } #endif diff --git a/cores/esp32/chip-debug-report.cpp b/cores/esp32/chip-debug-report.cpp index daafef3cab9..e3f3f431f81 100644 --- a/cores/esp32/chip-debug-report.cpp +++ b/cores/esp32/chip-debug-report.cpp @@ -88,6 +88,11 @@ static void printChipInfo(void) { case CHIP_ESP32C6: chip_report_printf("ESP32-C6\n"); break; case CHIP_ESP32H2: chip_report_printf("ESP32-H2\n"); break; case CHIP_ESP32P4: chip_report_printf("ESP32-P4\n"); break; +#if ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 5, 0) + case CHIP_ESP32C5: chip_report_printf("ESP32-C5\n"); break; + case CHIP_ESP32C61: chip_report_printf("ESP32-C61\n"); break; + case CHIP_ESP32H21: chip_report_printf("ESP32-H21\n"); break; +#endif default: chip_report_printf("Unknown %d\n", info.model); break; } printPkgVersion(); From 100ed8e89524d5228ea2fc8d5220eba775462519 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jan=20Proch=C3=A1zka?= <90197375+P-R-O-C-H-Y@users.noreply.github.com> Date: Wed, 22 Jan 2025 12:56:41 +0100 Subject: [PATCH 005/173] fix(c5): Update debug log in setCpuFrequencyMhz --- cores/esp32/esp32-hal-cpu.c | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/cores/esp32/esp32-hal-cpu.c b/cores/esp32/esp32-hal-cpu.c index 8c13a4077fd..30c7e3a2b0a 100644 --- a/cores/esp32/esp32-hal-cpu.c +++ b/cores/esp32/esp32-hal-cpu.c @@ -267,6 +267,12 @@ bool setCpuFrequencyMhz(uint32_t cpu_freq_mhz) { (conf.source == SOC_CPU_CLK_SRC_PLL) ? "PLL" : ((conf.source == SOC_CPU_CLK_SRC_APLL) ? "APLL" : ((conf.source == SOC_CPU_CLK_SRC_XTAL) ? "XTAL" : "8M")), conf.source_freq_mhz, conf.div, conf.freq_mhz, apb ); +#elif defined(CONFIG_IDF_TARGET_ESP32C5) + log_d( + "%s: %u / %u = %u Mhz, APB: %u Hz", + (conf.source == SOC_CPU_CLK_SRC_PLL_F240M || conf.source == SOC_CPU_CLK_SRC_PLL_F160M) ? "PLL" : ((conf.source == SOC_CPU_CLK_SRC_XTAL) ? "XTAL" : "8M"), + conf.source_freq_mhz, conf.div, conf.freq_mhz, apb + ); #else log_d( "%s: %u / %u = %u Mhz, APB: %u Hz", (conf.source == SOC_CPU_CLK_SRC_PLL) ? "PLL" : ((conf.source == SOC_CPU_CLK_SRC_XTAL) ? "XTAL" : "17.5M"), From d9d3bf48d3f25bd3bc8a2217c67034c5842eccb5 Mon Sep 17 00:00:00 2001 From: Jason2866 <24528715+Jason2866@users.noreply.github.com> Date: Wed, 22 Jan 2025 14:50:05 +0100 Subject: [PATCH 006/173] add c5 bootloader location to pioarduino script (#10889) * add c5 bootloader location to pioarduino script * ci(pre-commit): Apply automatic fixes --------- Co-authored-by: pre-commit-ci-lite[bot] <117423508+pre-commit-ci-lite[bot]@users.noreply.github.com> --- tools/pioarduino-build.py | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/tools/pioarduino-build.py b/tools/pioarduino-build.py index 3335a716888..47940b0d043 100644 --- a/tools/pioarduino-build.py +++ b/tools/pioarduino-build.py @@ -213,7 +213,11 @@ def add_tinyuf2_extra_image(): LIBSOURCE_DIRS=[join(FRAMEWORK_DIR, "libraries")], FLASH_EXTRA_IMAGES=[ ( - "0x1000" if build_mcu in ["esp32", "esp32s2"] else ("0x2000" if build_mcu in ["esp32p4"] else "0x0000"), + ( + "0x1000" + if build_mcu in ["esp32", "esp32s2"] + else ("0x2000" if build_mcu in ["esp32p4", "esp32c5"] else "0x0000") + ), get_bootloader_image(variants_dir), ), ("0x8000", join(env.subst("$BUILD_DIR"), "partitions.bin")), From 8af81cdf096f003255679485733566ffb23f3042 Mon Sep 17 00:00:00 2001 From: me-no-dev Date: Wed, 22 Jan 2025 15:52:13 +0200 Subject: [PATCH 007/173] fix(ci): Fix/stop some examples for C5 --- libraries/ESP32/examples/DeepSleep/ExternalWakeUp/ci.json | 3 ++- .../ESP32/examples/DeepSleep/SmoothBlink_ULP_Code/ci.json | 3 ++- libraries/ESP32/examples/DeepSleep/TouchWakeUp/ci.json | 3 ++- libraries/ESP32/examples/RMT/RMTLoopback/RMTLoopback.ino | 2 +- .../ESP32/examples/ResetReason/ResetReason/ResetReason.ino | 2 ++ libraries/ESP32/examples/TWAI/TWAIreceive/ci.json | 5 +++++ libraries/ESP32/examples/TWAI/TWAItransmit/ci.json | 5 +++++ libraries/ESP_SR/examples/Basic/ci.json | 3 ++- 8 files changed, 21 insertions(+), 5 deletions(-) create mode 100644 libraries/ESP32/examples/TWAI/TWAIreceive/ci.json create mode 100644 libraries/ESP32/examples/TWAI/TWAItransmit/ci.json diff --git a/libraries/ESP32/examples/DeepSleep/ExternalWakeUp/ci.json b/libraries/ESP32/examples/DeepSleep/ExternalWakeUp/ci.json index cd679adefad..dfd49d94fe9 100644 --- a/libraries/ESP32/examples/DeepSleep/ExternalWakeUp/ci.json +++ b/libraries/ESP32/examples/DeepSleep/ExternalWakeUp/ci.json @@ -3,6 +3,7 @@ "esp32c3": false, "esp32c6": false, "esp32h2": false, - "esp32p4": false + "esp32p4": false, + "esp32c5": false } } diff --git a/libraries/ESP32/examples/DeepSleep/SmoothBlink_ULP_Code/ci.json b/libraries/ESP32/examples/DeepSleep/SmoothBlink_ULP_Code/ci.json index 6afa60f44c4..5fa2bd14e5d 100644 --- a/libraries/ESP32/examples/DeepSleep/SmoothBlink_ULP_Code/ci.json +++ b/libraries/ESP32/examples/DeepSleep/SmoothBlink_ULP_Code/ci.json @@ -5,6 +5,7 @@ "esp32h2": false, "esp32p4": false, "esp32s2": false, - "esp32s3": false + "esp32s3": false, + "esp32c5": false } } diff --git a/libraries/ESP32/examples/DeepSleep/TouchWakeUp/ci.json b/libraries/ESP32/examples/DeepSleep/TouchWakeUp/ci.json index 25c42144223..ae65fa0df74 100644 --- a/libraries/ESP32/examples/DeepSleep/TouchWakeUp/ci.json +++ b/libraries/ESP32/examples/DeepSleep/TouchWakeUp/ci.json @@ -2,6 +2,7 @@ "targets": { "esp32c3": false, "esp32c6": false, - "esp32h2": false + "esp32h2": false, + "esp32c5": false } } diff --git a/libraries/ESP32/examples/RMT/RMTLoopback/RMTLoopback.ino b/libraries/ESP32/examples/RMT/RMTLoopback/RMTLoopback.ino index 17e7af290bf..1b96d868ecb 100644 --- a/libraries/ESP32/examples/RMT/RMTLoopback/RMTLoopback.ino +++ b/libraries/ESP32/examples/RMT/RMTLoopback/RMTLoopback.ino @@ -21,7 +21,7 @@ * */ -#if CONFIG_IDF_TARGET_ESP32C3 || CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 +#if CONFIG_IDF_TARGET_ESP32C3 || CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 || CONFIG_IDF_TARGET_ESP32C5 // ESP32 C3 has only 2 channels for RX and 2 for TX, thus MAX RMT_MEM is 128 #define RMT_TX_PIN 4 #define RMT_RX_PIN 5 diff --git a/libraries/ESP32/examples/ResetReason/ResetReason/ResetReason.ino b/libraries/ESP32/examples/ResetReason/ResetReason/ResetReason.ino index 0104c6422f2..ca7e15bf479 100644 --- a/libraries/ESP32/examples/ResetReason/ResetReason/ResetReason.ino +++ b/libraries/ESP32/examples/ResetReason/ResetReason/ResetReason.ino @@ -28,6 +28,8 @@ #include "esp32h2/rom/rtc.h" #elif CONFIG_IDF_TARGET_ESP32P4 #include "esp32p4/rom/rtc.h" +#elif CONFIG_IDF_TARGET_ESP32C5 +#include "esp32c5/rom/rtc.h" #else #error Target CONFIG_IDF_TARGET is not supported #endif diff --git a/libraries/ESP32/examples/TWAI/TWAIreceive/ci.json b/libraries/ESP32/examples/TWAI/TWAIreceive/ci.json new file mode 100644 index 00000000000..7379dba8bb9 --- /dev/null +++ b/libraries/ESP32/examples/TWAI/TWAIreceive/ci.json @@ -0,0 +1,5 @@ +{ + "targets": { + "esp32c5": false + } +} diff --git a/libraries/ESP32/examples/TWAI/TWAItransmit/ci.json b/libraries/ESP32/examples/TWAI/TWAItransmit/ci.json new file mode 100644 index 00000000000..7379dba8bb9 --- /dev/null +++ b/libraries/ESP32/examples/TWAI/TWAItransmit/ci.json @@ -0,0 +1,5 @@ +{ + "targets": { + "esp32c5": false + } +} diff --git a/libraries/ESP_SR/examples/Basic/ci.json b/libraries/ESP_SR/examples/Basic/ci.json index c395378f45e..ec0e969a7d0 100644 --- a/libraries/ESP_SR/examples/Basic/ci.json +++ b/libraries/ESP_SR/examples/Basic/ci.json @@ -13,6 +13,7 @@ "esp32c6": false, "esp32h2": false, "esp32p4": false, - "esp32s2": false + "esp32s2": false, + "esp32c5": false } } From 9fbcb345b786e30de491e34b1cd4ea8d8ff4e2ef Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jan=20Proch=C3=A1zka?= <90197375+P-R-O-C-H-Y@users.noreply.github.com> Date: Wed, 22 Jan 2025 18:44:49 +0100 Subject: [PATCH 008/173] fix(c5): Update PIN_RGB_LED in pins_arduino.h --- variants/esp32c5/pins_arduino.h | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/variants/esp32c5/pins_arduino.h b/variants/esp32c5/pins_arduino.h index eeaf3e01ebb..6ed7f0320fe 100644 --- a/variants/esp32c5/pins_arduino.h +++ b/variants/esp32c5/pins_arduino.h @@ -4,7 +4,7 @@ #include #include "soc/soc_caps.h" -#define PIN_RGB_LED 8 +#define PIN_RGB_LED 27 // BUILTIN_LED can be used in new Arduino API digitalWrite() like in Blink.ino static const uint8_t LED_BUILTIN = SOC_GPIO_PIN_COUNT + PIN_RGB_LED; #define BUILTIN_LED LED_BUILTIN // backward compatibility From f45cd7bf337316f4c6dde18bf23cd9c5b413863f Mon Sep 17 00:00:00 2001 From: me-no-dev Date: Mon, 27 Jan 2025 14:06:54 +0200 Subject: [PATCH 009/173] fix(psram): Add support for ESP32-C5 PSRAM --- boards.txt | 5 +++++ cores/esp32/chip-debug-report.cpp | 3 +++ cores/esp32/esp32-hal-psram.c | 2 ++ 3 files changed, 10 insertions(+) diff --git a/boards.txt b/boards.txt index af3e217df9d..0f42aa18a05 100644 --- a/boards.txt +++ b/boards.txt @@ -210,6 +210,11 @@ esp32c5.menu.JTAGAdapter.bridge=ESP USB Bridge esp32c5.menu.JTAGAdapter.bridge.build.openocdscript=esp32c5-bridge.cfg esp32c5.menu.JTAGAdapter.bridge.build.copy_jtag_files=1 +esp32c5.menu.PSRAM.disabled=Disabled +esp32c5.menu.PSRAM.disabled.build.defines= +esp32c5.menu.PSRAM.enabled=Enabled +esp32c5.menu.PSRAM.enabled.build.defines=-DBOARD_HAS_PSRAM + esp32c5.menu.CDCOnBoot.default=Disabled esp32c5.menu.CDCOnBoot.default.build.cdc_on_boot=0 esp32c5.menu.CDCOnBoot.cdc=Enabled diff --git a/cores/esp32/chip-debug-report.cpp b/cores/esp32/chip-debug-report.cpp index e3f3f431f81..281c7bdb62d 100644 --- a/cores/esp32/chip-debug-report.cpp +++ b/cores/esp32/chip-debug-report.cpp @@ -67,6 +67,9 @@ static void printPkgVersion(void) { #elif CONFIG_IDF_TARGET_ESP32P4 uint32_t pkg_ver = REG_GET_FIELD(EFUSE_RD_MAC_SYS_2_REG, EFUSE_PKG_VERSION); chip_report_printf("%lu", pkg_ver); +#elif CONFIG_IDF_TARGET_ESP32C5 + uint32_t pkg_ver = REG_GET_FIELD(EFUSE_RD_MAC_SYS2_REG, EFUSE_PKG_VERSION); + chip_report_printf("%lu", pkg_ver); #else chip_report_printf("Unknown"); #endif diff --git a/cores/esp32/esp32-hal-psram.c b/cores/esp32/esp32-hal-psram.c index 3c7a51c3343..0d57a67ede4 100644 --- a/cores/esp32/esp32-hal-psram.c +++ b/cores/esp32/esp32-hal-psram.c @@ -29,6 +29,8 @@ #include "esp32s3/rom/cache.h" #elif CONFIG_IDF_TARGET_ESP32P4 #include "esp32p4/rom/cache.h" +#elif CONFIG_IDF_TARGET_ESP32C5 +#include "esp32c5/rom/cache.h" #else #error Target CONFIG_IDF_TARGET is not supported #endif From 75de09ef298c5bc9e26c7be2fd8af0446b4e4296 Mon Sep 17 00:00:00 2001 From: me-no-dev Date: Tue, 4 Feb 2025 00:14:19 +0200 Subject: [PATCH 010/173] fix(board): Update ESP32-C5 Dev Kit Pinout --- variants/esp32c5/pins_arduino.h | 24 +++++++++++++++--------- 1 file changed, 15 insertions(+), 9 deletions(-) diff --git a/variants/esp32c5/pins_arduino.h b/variants/esp32c5/pins_arduino.h index 6ed7f0320fe..e3a7e57ef44 100644 --- a/variants/esp32c5/pins_arduino.h +++ b/variants/esp32c5/pins_arduino.h @@ -19,13 +19,13 @@ static const uint8_t RX = 12; // static const uint8_t USB_DM = 13; // static const uint8_t USB_DP = 14; -static const uint8_t SDA = 23; -static const uint8_t SCL = 22; +static const uint8_t SDA = 0; +static const uint8_t SCL = 1; -static const uint8_t SS = 18; -static const uint8_t MOSI = 19; -static const uint8_t MISO = 20; -static const uint8_t SCK = 21; +static const uint8_t SS = 6; +static const uint8_t MOSI = 8; +static const uint8_t MISO = 9; +static const uint8_t SCK = 10; static const uint8_t A0 = 1; static const uint8_t A1 = 2; @@ -34,9 +34,15 @@ static const uint8_t A3 = 4; static const uint8_t A4 = 5; static const uint8_t A5 = 6; -// LP I2C Pins are fixed on ESP32-C6 +// LP I2C Pins are fixed on ESP32-C5 +static const uint8_t LP_SDA = 2; +static const uint8_t LP_SCL = 3; #define WIRE1_PIN_DEFINED -static const uint8_t SDA1 = 2; -static const uint8_t SCL1 = 3; +#define SDA1 LP_SDA +#define SCL1 LP_SCL + +// LP UART Pins are fixed on ESP32-C5 +static const uint8_t LP_RX = 4; +static const uint8_t LP_TX = 5; #endif /* Pins_Arduino_h */ From 6283c15ae22da9ec731b2fc7a1f5539f9efca6cf Mon Sep 17 00:00:00 2001 From: Me No Dev Date: Tue, 18 Feb 2025 14:24:37 +0200 Subject: [PATCH 011/173] IDF master (#10887) * feat(ci): Run sketches on ESP32-C5 * IDF master 1160a86b * fix(zigbee): Remove RCP mode from ESP32-C5 --- .github/scripts/on-push.sh | 1 + .github/scripts/sketch_utils.sh | 5 + .github/workflows/push.yml | 1 + boards.txt | 14 +- package/package_esp32_index.template.json | 204 +++++++++++----------- 5 files changed, 113 insertions(+), 112 deletions(-) diff --git a/.github/scripts/on-push.sh b/.github/scripts/on-push.sh index 6095f88e727..3eb020c09a3 100755 --- a/.github/scripts/on-push.sh +++ b/.github/scripts/on-push.sh @@ -90,6 +90,7 @@ if [ "$BUILD_LOG" -eq 1 ]; then fi #build sketches for different targets +build "esp32c5" "$CHUNK_INDEX" "$CHUNKS_CNT" "$BUILD_LOG" "$LOG_LEVEL" "$SKETCHES_FILE" "${SKETCHES_ESP32[@]}" build "esp32p4" "$CHUNK_INDEX" "$CHUNKS_CNT" "$BUILD_LOG" "$LOG_LEVEL" "$SKETCHES_FILE" "${SKETCHES_ESP32[@]}" build "esp32s3" "$CHUNK_INDEX" "$CHUNKS_CNT" "$BUILD_LOG" "$LOG_LEVEL" "$SKETCHES_FILE" "${SKETCHES_ESP32[@]}" build "esp32s2" "$CHUNK_INDEX" "$CHUNKS_CNT" "$BUILD_LOG" "$LOG_LEVEL" "$SKETCHES_FILE" "${SKETCHES_ESP32[@]}" diff --git a/.github/scripts/sketch_utils.sh b/.github/scripts/sketch_utils.sh index e536da50111..323f39b3afc 100755 --- a/.github/scripts/sketch_utils.sh +++ b/.github/scripts/sketch_utils.sh @@ -156,6 +156,7 @@ function build_sketch { # build_sketch [ext esp32c6_opts=$(echo "$debug_level,$fqbn_append" | sed 's/^,*//;s/,*$//;s/,\{2,\}/,/g') esp32h2_opts=$(echo "$debug_level,$fqbn_append" | sed 's/^,*//;s/,*$//;s/,\{2,\}/,/g') esp32p4_opts=$(echo "PSRAM=enabled,USBMode=default,$debug_level,$fqbn_append" | sed 's/^,*//;s/,*$//;s/,\{2,\}/,/g') + esp32c5_opts=$(echo "$debug_level,$fqbn_append" | sed 's/^,*//;s/,*$//;s/,\{2,\}/,/g') # Select the common part of the FQBN based on the target. The rest will be # appended depending on the passed options. @@ -191,6 +192,10 @@ function build_sketch { # build_sketch [ext [ -n "${options:-$esp32p4_opts}" ] && opt=":${options:-$esp32p4_opts}" fqbn="espressif:esp32:esp32p4$opt" ;; + "esp32c5") + [ -n "${options:-$esp32c5_opts}" ] && opt=":${options:-$esp32c5_opts}" + fqbn="espressif:esp32:esp32c5$opt" + ;; *) echo "ERROR: Invalid chip: $target" exit 1 diff --git a/.github/workflows/push.yml b/.github/workflows/push.yml index feb32f95d03..428b89a8c21 100644 --- a/.github/workflows/push.yml +++ b/.github/workflows/push.yml @@ -47,6 +47,7 @@ on: - "variants/esp32/**/*" - "variants/esp32c2/**/*" - "variants/esp32c3/**/*" + - "variants/esp32c5/**/*" - "variants/esp32c6/**/*" - "variants/esp32h2/**/*" - "variants/esp32p4/**/*" diff --git a/boards.txt b/boards.txt index 0f42aa18a05..f7ad60d3eab 100644 --- a/boards.txt +++ b/boards.txt @@ -359,22 +359,16 @@ esp32c5.menu.ZigbeeMode.default.build.zigbee_mode= esp32c5.menu.ZigbeeMode.default.build.zigbee_libs= esp32c5.menu.ZigbeeMode.ed=Zigbee ED (end device) esp32c5.menu.ZigbeeMode.ed.build.zigbee_mode=-DZIGBEE_MODE_ED -esp32c5.menu.ZigbeeMode.ed.build.zigbee_libs=-lesp_zb_api_ed -lesp_zb_cli_command -lzboss_stack.ed -lzboss_port +esp32c5.menu.ZigbeeMode.ed.build.zigbee_libs=-lesp_zb_api.ed -lzboss_stack.ed -lzboss_port.native esp32c5.menu.ZigbeeMode.zczr=Zigbee ZCZR (coordinator/router) esp32c5.menu.ZigbeeMode.zczr.build.zigbee_mode=-DZIGBEE_MODE_ZCZR -esp32c5.menu.ZigbeeMode.zczr.build.zigbee_libs=-lesp_zb_api_zczr -lesp_zb_cli_command -lzboss_stack.zczr -lzboss_port -esp32c5.menu.ZigbeeMode.rcp=Zigbee RCP (radio co-processor) -esp32c5.menu.ZigbeeMode.rcp.build.zigbee_mode=-DZIGBEE_MODE_RCP -esp32c5.menu.ZigbeeMode.rcp.build.zigbee_libs=-lesp_zb_api_rcp -lesp_zb_cli_command -lzboss_stack.rcp -lzboss_port +esp32c5.menu.ZigbeeMode.zczr.build.zigbee_libs=-lesp_zb_api.zczr -lzboss_stack.zczr -lzboss_port.native esp32c5.menu.ZigbeeMode.ed_debug=Zigbee ED (end device) - Debug esp32c5.menu.ZigbeeMode.ed_debug.build.zigbee_mode=-DZIGBEE_MODE_ED -esp32c5.menu.ZigbeeMode.ed_debug.build.zigbee_libs=-lesp_zb_api_ed.debug -lesp_zb_cli_command -lzboss_stack.ed.debug -lzboss_port.debug +esp32c5.menu.ZigbeeMode.ed_debug.build.zigbee_libs=-lesp_zb_api.ed.debug -lzboss_stack.ed.debug -lzboss_port.native.debug esp32c5.menu.ZigbeeMode.zczr_debug=Zigbee ZCZR (coordinator/router) - Debug esp32c5.menu.ZigbeeMode.zczr_debug.build.zigbee_mode=-DZIGBEE_MODE_ZCZR -esp32c5.menu.ZigbeeMode.zczr_debug.build.zigbee_libs=-lesp_zb_api_zczr.debug -lesp_zb_cli_command -lzboss_stack.zczr.debug -lzboss_port.debug -esp32c5.menu.ZigbeeMode.rcp_debug=Zigbee RCP (radio co-processor) - Debug -esp32c5.menu.ZigbeeMode.rcp_debug.build.zigbee_mode=-DZIGBEE_MODE_RCP -esp32c5.menu.ZigbeeMode.rcp_debug.build.zigbee_libs=-lesp_zb_api_rcp.debug -lesp_zb_cli_command -lzboss_stack.rcp.debug -lzboss_port.debug +esp32c5.menu.ZigbeeMode.zczr_debug.build.zigbee_libs=-lesp_zb_api.zczr.debug -lzboss_stack.zczr.debug -lzboss_port.native.debug ############################################################## diff --git a/package/package_esp32_index.template.json b/package/package_esp32_index.template.json index b0d7c3200d1..8bbd2e6a4e9 100644 --- a/package/package_esp32_index.template.json +++ b/package/package_esp32_index.template.json @@ -51,7 +51,7 @@ { "packager": "esp32", "name": "esp32-arduino-libs", - "version": "idf-release_v5.4-d4aa25a3-v1" + "version": "idf-master-1160a86b-v1" }, { "packager": "esp32", @@ -61,7 +61,7 @@ { "packager": "esp32", "name": "xtensa-esp-elf-gdb", - "version": "14.2_20240403" + "version": "15.2_20241112" }, { "packager": "esp32", @@ -71,7 +71,7 @@ { "packager": "esp32", "name": "riscv32-esp-elf-gdb", - "version": "14.2_20240403" + "version": "15.2_20241112" }, { "packager": "esp32", @@ -104,63 +104,63 @@ "tools": [ { "name": "esp32-arduino-libs", - "version": "idf-release_v5.4-d4aa25a3-v1", + "version": "idf-master-1160a86b-v1", "systems": [ { "host": "i686-mingw32", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.4/esp32-arduino-libs-idf-release_v5.4-d4aa25a3-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.4-d4aa25a3-v1.zip", - "checksum": "SHA-256:81101d580ebafb78f71bd494f4f5162fd829279d18634282c0f8f95c9e928335", - "size": "350941396" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-1160a86b-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-1160a86b-v1.zip", + "checksum": "SHA-256:ab71bb61b12ad109750d4d39200c2aeaa826395f65a2c95c2012d969eb5f76a2", + "size": "405894668" }, { "host": "x86_64-mingw32", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.4/esp32-arduino-libs-idf-release_v5.4-d4aa25a3-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.4-d4aa25a3-v1.zip", - "checksum": "SHA-256:81101d580ebafb78f71bd494f4f5162fd829279d18634282c0f8f95c9e928335", - "size": "350941396" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-1160a86b-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-1160a86b-v1.zip", + "checksum": "SHA-256:ab71bb61b12ad109750d4d39200c2aeaa826395f65a2c95c2012d969eb5f76a2", + "size": "405894668" }, { "host": "arm64-apple-darwin", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.4/esp32-arduino-libs-idf-release_v5.4-d4aa25a3-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.4-d4aa25a3-v1.zip", - "checksum": "SHA-256:81101d580ebafb78f71bd494f4f5162fd829279d18634282c0f8f95c9e928335", - "size": "350941396" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-1160a86b-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-1160a86b-v1.zip", + "checksum": "SHA-256:ab71bb61b12ad109750d4d39200c2aeaa826395f65a2c95c2012d969eb5f76a2", + "size": "405894668" }, { "host": "x86_64-apple-darwin", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.4/esp32-arduino-libs-idf-release_v5.4-d4aa25a3-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.4-d4aa25a3-v1.zip", - "checksum": "SHA-256:81101d580ebafb78f71bd494f4f5162fd829279d18634282c0f8f95c9e928335", - "size": "350941396" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-1160a86b-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-1160a86b-v1.zip", + "checksum": "SHA-256:ab71bb61b12ad109750d4d39200c2aeaa826395f65a2c95c2012d969eb5f76a2", + "size": "405894668" }, { "host": "x86_64-pc-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.4/esp32-arduino-libs-idf-release_v5.4-d4aa25a3-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.4-d4aa25a3-v1.zip", - "checksum": "SHA-256:81101d580ebafb78f71bd494f4f5162fd829279d18634282c0f8f95c9e928335", - "size": "350941396" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-1160a86b-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-1160a86b-v1.zip", + "checksum": "SHA-256:ab71bb61b12ad109750d4d39200c2aeaa826395f65a2c95c2012d969eb5f76a2", + "size": "405894668" }, { "host": "i686-pc-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.4/esp32-arduino-libs-idf-release_v5.4-d4aa25a3-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.4-d4aa25a3-v1.zip", - "checksum": "SHA-256:81101d580ebafb78f71bd494f4f5162fd829279d18634282c0f8f95c9e928335", - "size": "350941396" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-1160a86b-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-1160a86b-v1.zip", + "checksum": "SHA-256:ab71bb61b12ad109750d4d39200c2aeaa826395f65a2c95c2012d969eb5f76a2", + "size": "405894668" }, { "host": "aarch64-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.4/esp32-arduino-libs-idf-release_v5.4-d4aa25a3-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.4-d4aa25a3-v1.zip", - "checksum": "SHA-256:81101d580ebafb78f71bd494f4f5162fd829279d18634282c0f8f95c9e928335", - "size": "350941396" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-1160a86b-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-1160a86b-v1.zip", + "checksum": "SHA-256:ab71bb61b12ad109750d4d39200c2aeaa826395f65a2c95c2012d969eb5f76a2", + "size": "405894668" }, { "host": "arm-linux-gnueabihf", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.4/esp32-arduino-libs-idf-release_v5.4-d4aa25a3-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.4-d4aa25a3-v1.zip", - "checksum": "SHA-256:81101d580ebafb78f71bd494f4f5162fd829279d18634282c0f8f95c9e928335", - "size": "350941396" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-1160a86b-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-1160a86b-v1.zip", + "checksum": "SHA-256:ab71bb61b12ad109750d4d39200c2aeaa826395f65a2c95c2012d969eb5f76a2", + "size": "405894668" } ] }, @@ -228,63 +228,63 @@ }, { "name": "xtensa-esp-elf-gdb", - "version": "14.2_20240403", + "version": "15.2_20241112", "systems": [ { "host": "x86_64-pc-linux-gnu", - "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v14.2_20240403/xtensa-esp-elf-gdb-14.2_20240403-x86_64-linux-gnu.tar.gz", - "archiveFileName": "xtensa-esp-elf-gdb-14.2_20240403-x86_64-linux-gnu.tar.gz", - "checksum": "SHA-256:9d68472d4cba5cf8c2b79d94f86f92c828e76a632bd1e6be5e7706e5b304d36e", - "size": "31010320" + "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v15.2_20241112/xtensa-esp-elf-gdb-15.2_20241112-x86_64-linux-gnu.tar.gz", + "archiveFileName": "xtensa-esp-elf-gdb-15.2_20241112-x86_64-linux-gnu.tar.gz", + "checksum": "SHA-256:18774349d2b1c7d7f5ba984563f7022aef4a3df4b706ed8821c53266f599343d", + "size": "35179121" }, { "host": "aarch64-linux-gnu", - "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v14.2_20240403/xtensa-esp-elf-gdb-14.2_20240403-aarch64-linux-gnu.tar.gz", - "archiveFileName": "xtensa-esp-elf-gdb-14.2_20240403-aarch64-linux-gnu.tar.gz", - "checksum": "SHA-256:bdabc3217994815fc311c4e16e588b78f6596b5ad4ffa46c80b40e982cfb1e66", - "size": "30954580" + "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v15.2_20241112/xtensa-esp-elf-gdb-15.2_20241112-aarch64-linux-gnu.tar.gz", + "archiveFileName": "xtensa-esp-elf-gdb-15.2_20241112-aarch64-linux-gnu.tar.gz", + "checksum": "SHA-256:77cb3b2c85d6cfbb40b7f99eebbc2b1c3f4fe13eba20b3da798bdbbc6eb8e87c", + "size": "34295046" }, { "host": "arm-linux-gnueabihf", - "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v14.2_20240403/xtensa-esp-elf-gdb-14.2_20240403-arm-linux-gnueabi.tar.gz", - "archiveFileName": "xtensa-esp-elf-gdb-14.2_20240403-arm-linux-gnueabi.tar.gz", - "checksum": "SHA-256:d54b8d703ba897b28c627da3d27106a3906dd01ba298778a67064710bc33c76d", - "size": "28697281" + "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v15.2_20241112/xtensa-esp-elf-gdb-15.2_20241112-arm-linux-gnueabi.tar.gz", + "archiveFileName": "xtensa-esp-elf-gdb-15.2_20241112-arm-linux-gnueabi.tar.gz", + "checksum": "SHA-256:b87af0539de118eb9d43a4a89c8c1b0a6ab2557560015155104c44f91b5c4aa0", + "size": "30338727" }, { "host": "i686-pc-linux-gnu", - "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v14.2_20240403/xtensa-esp-elf-gdb-14.2_20240403-i586-linux-gnu.tar.gz", - "archiveFileName": "xtensa-esp-elf-gdb-14.2_20240403-i586-linux-gnu.tar.gz", - "checksum": "SHA-256:64d3bc992ed8fdec383d49e8b803ac494605a38117c8293db8da055037de96b0", - "size": "29890994" + "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v15.2_20241112/xtensa-esp-elf-gdb-15.2_20241112-i586-linux-gnu.tar.gz", + "archiveFileName": "xtensa-esp-elf-gdb-15.2_20241112-i586-linux-gnu.tar.gz", + "checksum": "SHA-256:2c7531bd390928fed479999ac9089b39a25f56ca4f4cc330db7d7a633a37405b", + "size": "33906121" }, { "host": "x86_64-apple-darwin", - "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v14.2_20240403/xtensa-esp-elf-gdb-14.2_20240403-x86_64-apple-darwin14.tar.gz", - "archiveFileName": "xtensa-esp-elf-gdb-14.2_20240403-x86_64-apple-darwin14.tar.gz", - "checksum": "SHA-256:023e74b3fda793da4bc0509b02de776ee0dad6efaaac17bef5916fb7dc9c26b9", - "size": "44446611" + "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v15.2_20241112/xtensa-esp-elf-gdb-15.2_20241112-x86_64-apple-darwin21.1.tar.gz", + "archiveFileName": "xtensa-esp-elf-gdb-15.2_20241112-x86_64-apple-darwin21.1.tar.gz", + "checksum": "SHA-256:ba2907be9a4c22c4e418f42ec84cf57401570a71dabbe69425227114ebf351a7", + "size": "52502302" }, { "host": "arm64-apple-darwin", - "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v14.2_20240403/xtensa-esp-elf-gdb-14.2_20240403-aarch64-apple-darwin21.1.tar.gz", - "archiveFileName": "xtensa-esp-elf-gdb-14.2_20240403-aarch64-apple-darwin21.1.tar.gz", - "checksum": "SHA-256:ea757c6bf8c25238f6d2fdcc6bbab25a1b00608a0f9e19b7ddd2f37ddbdc3fb1", - "size": "37021423" + "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v15.2_20241112/xtensa-esp-elf-gdb-15.2_20241112-aarch64-apple-darwin21.1.tar.gz", + "archiveFileName": "xtensa-esp-elf-gdb-15.2_20241112-aarch64-apple-darwin21.1.tar.gz", + "checksum": "SHA-256:893500d6de354a6870820b9398531d8cd37d1cd85f3ae9b1f8a4c070b8048707", + "size": "41892363" }, { "host": "i686-mingw32", - "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v14.2_20240403/xtensa-esp-elf-gdb-14.2_20240403-i686-w64-mingw32.zip", - "archiveFileName": "xtensa-esp-elf-gdb-14.2_20240403-i686-w64-mingw32.zip", - "checksum": "SHA-256:322e8d9b700dc32d8158e3dc55fb85ec55de48d0bb7789375ee39a28d5d655e2", - "size": "26302466" + "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v15.2_20241112/xtensa-esp-elf-gdb-15.2_20241112-i686-w64-mingw32.zip", + "archiveFileName": "xtensa-esp-elf-gdb-15.2_20241112-i686-w64-mingw32.zip", + "checksum": "SHA-256:328181380fccb252105c51a86071edbc5ef63e0114540edc4f8b2d62acf44916", + "size": "31307770" }, { "host": "x86_64-mingw32", - "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v14.2_20240403/xtensa-esp-elf-gdb-14.2_20240403-x86_64-w64-mingw32.zip", - "archiveFileName": "xtensa-esp-elf-gdb-14.2_20240403-x86_64-w64-mingw32.zip", - "checksum": "SHA-256:a27a2fe20f192f8e0a51b8936428b4e1cf8935cfe008ee445cc49f6fc7f6db2e", - "size": "28366035" + "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v15.2_20241112/xtensa-esp-elf-gdb-15.2_20241112-x86_64-w64-mingw32.zip", + "archiveFileName": "xtensa-esp-elf-gdb-15.2_20241112-x86_64-w64-mingw32.zip", + "checksum": "SHA-256:9c1949058d7aa1fa6f6f2d03173659b9f54e3c3940cbeebc1d56ae169c604ab2", + "size": "31420687" } ] }, @@ -352,63 +352,63 @@ }, { "name": "riscv32-esp-elf-gdb", - "version": "14.2_20240403", + "version": "15.2_20241112", "systems": [ { "host": "x86_64-pc-linux-gnu", - "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v14.2_20240403/riscv32-esp-elf-gdb-14.2_20240403-x86_64-linux-gnu.tar.gz", - "archiveFileName": "riscv32-esp-elf-gdb-14.2_20240403-x86_64-linux-gnu.tar.gz", - "checksum": "SHA-256:ce004bc0bbd71b246800d2d13b239218b272a38bd528e316f21f1af2db8a4b13", - "size": "30707431" + "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v15.2_20241112/riscv32-esp-elf-gdb-15.2_20241112-x86_64-linux-gnu.tar.gz", + "archiveFileName": "riscv32-esp-elf-gdb-15.2_20241112-x86_64-linux-gnu.tar.gz", + "checksum": "SHA-256:bfca245b3d84244ad3b6156496728d916ac5ccc0d7f8e048194b7eba5cdbe047", + "size": "35297398" }, { "host": "aarch64-linux-gnu", - "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v14.2_20240403/riscv32-esp-elf-gdb-14.2_20240403-aarch64-linux-gnu.tar.gz", - "archiveFileName": "riscv32-esp-elf-gdb-14.2_20240403-aarch64-linux-gnu.tar.gz", - "checksum": "SHA-256:ba10f2866c61410b88c65957274280b1a62e3bed05131654ed9b6758efe18e55", - "size": "30824065" + "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v15.2_20241112/riscv32-esp-elf-gdb-15.2_20241112-aarch64-linux-gnu.tar.gz", + "archiveFileName": "riscv32-esp-elf-gdb-15.2_20241112-aarch64-linux-gnu.tar.gz", + "checksum": "SHA-256:8536da9e3093b8f25e0b5204b04ed4afea432d1fd262f2abb466016d3d750ea6", + "size": "34455317" }, { "host": "arm-linux-gnueabihf", - "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v14.2_20240403/riscv32-esp-elf-gdb-14.2_20240403-arm-linux-gnueabi.tar.gz", - "archiveFileName": "riscv32-esp-elf-gdb-14.2_20240403-arm-linux-gnueabi.tar.gz", - "checksum": "SHA-256:88539db5d987f28827efac7e26080a2803b9b539342ccd2963ccfdd56d7f08f7", - "size": "29000575" + "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v15.2_20241112/riscv32-esp-elf-gdb-15.2_20241112-arm-linux-gnueabi.tar.gz", + "archiveFileName": "riscv32-esp-elf-gdb-15.2_20241112-arm-linux-gnueabi.tar.gz", + "checksum": "SHA-256:33a80d5e6604bb7d08b15492b8ec84c9176245e066d0bf8aad7570786ca44081", + "size": "31188203" }, { "host": "i686-pc-linux-gnu", - "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v14.2_20240403/riscv32-esp-elf-gdb-14.2_20240403-i586-linux-gnu.tar.gz", - "archiveFileName": "riscv32-esp-elf-gdb-14.2_20240403-i586-linux-gnu.tar.gz", - "checksum": "SHA-256:0e628ee37438ab6ba05eb889a76d09e50cb98e0020a16b8e2b935c5cf19b4ed2", - "size": "29947521" + "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v15.2_20241112/riscv32-esp-elf-gdb-15.2_20241112-i586-linux-gnu.tar.gz", + "archiveFileName": "riscv32-esp-elf-gdb-15.2_20241112-i586-linux-gnu.tar.gz", + "checksum": "SHA-256:ba448ecf2c80064013eaacb0e7be514a5ef17516f49ff3e50bc41c5578b8d88e", + "size": "34211289" }, { "host": "x86_64-apple-darwin", - "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v14.2_20240403/riscv32-esp-elf-gdb-14.2_20240403-x86_64-apple-darwin14.tar.gz", - "archiveFileName": "riscv32-esp-elf-gdb-14.2_20240403-x86_64-apple-darwin14.tar.gz", - "checksum": "SHA-256:8f6bda832d70dad5860a639d55aba4237bd10cbac9f4822db1eece97357b34a9", - "size": "44196117" + "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v15.2_20241112/riscv32-esp-elf-gdb-15.2_20241112-x86_64-apple-darwin21.1.tar.gz", + "archiveFileName": "riscv32-esp-elf-gdb-15.2_20241112-x86_64-apple-darwin21.1.tar.gz", + "checksum": "SHA-256:bdd07c54fe3216eb5c34f92cac514eb4af777c951573f186c33408c75f56bb4b", + "size": "52831957" }, { "host": "arm64-apple-darwin", - "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v14.2_20240403/riscv32-esp-elf-gdb-14.2_20240403-aarch64-apple-darwin21.1.tar.gz", - "archiveFileName": "riscv32-esp-elf-gdb-14.2_20240403-aarch64-apple-darwin21.1.tar.gz", - "checksum": "SHA-256:d88b6116e86456c8480ce9bc95aed375a35c0d091f1da0a53b86be0e6ef3d320", - "size": "36794404" + "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v15.2_20241112/riscv32-esp-elf-gdb-15.2_20241112-aarch64-apple-darwin21.1.tar.gz", + "archiveFileName": "riscv32-esp-elf-gdb-15.2_20241112-aarch64-apple-darwin21.1.tar.gz", + "checksum": "SHA-256:9924f439ae77c3346e532a48a0d56330466c33ce1abd8d72b77f9f7f5c3b1196", + "size": "42227631" }, { "host": "i686-mingw32", - "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v14.2_20240403/riscv32-esp-elf-gdb-14.2_20240403-i686-w64-mingw32.zip", - "archiveFileName": "riscv32-esp-elf-gdb-14.2_20240403-i686-w64-mingw32.zip", - "checksum": "SHA-256:d6e7ce05805b0d8d4dd138ad239b98a1adf8da98941867d60760eb1ae5361730", - "size": "26486295" + "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v15.2_20241112/riscv32-esp-elf-gdb-15.2_20241112-i686-w64-mingw32.zip", + "archiveFileName": "riscv32-esp-elf-gdb-15.2_20241112-i686-w64-mingw32.zip", + "checksum": "SHA-256:dcdafd30854b092671f555d844d94b0733e2ffcac8c026b4fb7b8a72ebef0016", + "size": "31971712" }, { "host": "x86_64-mingw32", - "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v14.2_20240403/riscv32-esp-elf-gdb-14.2_20240403-x86_64-w64-mingw32.zip", - "archiveFileName": "riscv32-esp-elf-gdb-14.2_20240403-x86_64-w64-mingw32.zip", - "checksum": "SHA-256:5c9f211dc46daf6b96fad09d709284a0f0186fef8947d9f6edd6bca5b5ad4317", - "size": "27942579" + "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v15.2_20241112/riscv32-esp-elf-gdb-15.2_20241112-x86_64-w64-mingw32.zip", + "archiveFileName": "riscv32-esp-elf-gdb-15.2_20241112-x86_64-w64-mingw32.zip", + "checksum": "SHA-256:6ec8b3a073f2c5835321b4b560f8fc56180458204d071da139ff983436ea45e5", + "size": "31759225" } ] }, From 042015efc8930c3de0633ad930c8c05ed536aadd Mon Sep 17 00:00:00 2001 From: Me No Dev Date: Tue, 18 Feb 2025 21:09:27 +0200 Subject: [PATCH 012/173] IDF master (#10981) * IDF master 1160a86b * fix(ci): Do not compile RainMaker examples on ESP32 --- .../RainMaker/examples/RMakerCustom/ci.json | 3 + .../RainMaker/examples/RMakerSwitch/ci.json | 3 + package/package_esp32_index.template.json | 68 +++++++++---------- 3 files changed, 40 insertions(+), 34 deletions(-) diff --git a/libraries/RainMaker/examples/RMakerCustom/ci.json b/libraries/RainMaker/examples/RMakerCustom/ci.json index 1c80eda1d90..ce63fe9ccf0 100644 --- a/libraries/RainMaker/examples/RMakerCustom/ci.json +++ b/libraries/RainMaker/examples/RMakerCustom/ci.json @@ -1,4 +1,7 @@ { + "targets": { + "esp32": false + }, "fqbn_append": "PartitionScheme=rainmaker_4MB", "requires": [ "CONFIG_ESP_RMAKER_WORK_QUEUE_TASK_STACK=[1-9][0-9]*" diff --git a/libraries/RainMaker/examples/RMakerSwitch/ci.json b/libraries/RainMaker/examples/RMakerSwitch/ci.json index 1c80eda1d90..ce63fe9ccf0 100644 --- a/libraries/RainMaker/examples/RMakerSwitch/ci.json +++ b/libraries/RainMaker/examples/RMakerSwitch/ci.json @@ -1,4 +1,7 @@ { + "targets": { + "esp32": false + }, "fqbn_append": "PartitionScheme=rainmaker_4MB", "requires": [ "CONFIG_ESP_RMAKER_WORK_QUEUE_TASK_STACK=[1-9][0-9]*" diff --git a/package/package_esp32_index.template.json b/package/package_esp32_index.template.json index 8bbd2e6a4e9..714328f9a17 100644 --- a/package/package_esp32_index.template.json +++ b/package/package_esp32_index.template.json @@ -51,7 +51,7 @@ { "packager": "esp32", "name": "esp32-arduino-libs", - "version": "idf-master-1160a86b-v1" + "version": "idf-master-1160a86b-v2" }, { "packager": "esp32", @@ -104,63 +104,63 @@ "tools": [ { "name": "esp32-arduino-libs", - "version": "idf-master-1160a86b-v1", + "version": "idf-master-1160a86b-v2", "systems": [ { "host": "i686-mingw32", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-1160a86b-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-1160a86b-v1.zip", - "checksum": "SHA-256:ab71bb61b12ad109750d4d39200c2aeaa826395f65a2c95c2012d969eb5f76a2", - "size": "405894668" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-1160a86b-v2.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-1160a86b-v2.zip", + "checksum": "SHA-256:2050bda30b1d3cc397683f38267744dc45d7e4c4dc857250477117f30db754db", + "size": "406007754" }, { "host": "x86_64-mingw32", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-1160a86b-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-1160a86b-v1.zip", - "checksum": "SHA-256:ab71bb61b12ad109750d4d39200c2aeaa826395f65a2c95c2012d969eb5f76a2", - "size": "405894668" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-1160a86b-v2.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-1160a86b-v2.zip", + "checksum": "SHA-256:2050bda30b1d3cc397683f38267744dc45d7e4c4dc857250477117f30db754db", + "size": "406007754" }, { "host": "arm64-apple-darwin", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-1160a86b-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-1160a86b-v1.zip", - "checksum": "SHA-256:ab71bb61b12ad109750d4d39200c2aeaa826395f65a2c95c2012d969eb5f76a2", - "size": "405894668" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-1160a86b-v2.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-1160a86b-v2.zip", + "checksum": "SHA-256:2050bda30b1d3cc397683f38267744dc45d7e4c4dc857250477117f30db754db", + "size": "406007754" }, { "host": "x86_64-apple-darwin", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-1160a86b-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-1160a86b-v1.zip", - "checksum": "SHA-256:ab71bb61b12ad109750d4d39200c2aeaa826395f65a2c95c2012d969eb5f76a2", - "size": "405894668" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-1160a86b-v2.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-1160a86b-v2.zip", + "checksum": "SHA-256:2050bda30b1d3cc397683f38267744dc45d7e4c4dc857250477117f30db754db", + "size": "406007754" }, { "host": "x86_64-pc-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-1160a86b-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-1160a86b-v1.zip", - "checksum": "SHA-256:ab71bb61b12ad109750d4d39200c2aeaa826395f65a2c95c2012d969eb5f76a2", - "size": "405894668" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-1160a86b-v2.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-1160a86b-v2.zip", + "checksum": "SHA-256:2050bda30b1d3cc397683f38267744dc45d7e4c4dc857250477117f30db754db", + "size": "406007754" }, { "host": "i686-pc-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-1160a86b-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-1160a86b-v1.zip", - "checksum": "SHA-256:ab71bb61b12ad109750d4d39200c2aeaa826395f65a2c95c2012d969eb5f76a2", - "size": "405894668" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-1160a86b-v2.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-1160a86b-v2.zip", + "checksum": "SHA-256:2050bda30b1d3cc397683f38267744dc45d7e4c4dc857250477117f30db754db", + "size": "406007754" }, { "host": "aarch64-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-1160a86b-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-1160a86b-v1.zip", - "checksum": "SHA-256:ab71bb61b12ad109750d4d39200c2aeaa826395f65a2c95c2012d969eb5f76a2", - "size": "405894668" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-1160a86b-v2.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-1160a86b-v2.zip", + "checksum": "SHA-256:2050bda30b1d3cc397683f38267744dc45d7e4c4dc857250477117f30db754db", + "size": "406007754" }, { "host": "arm-linux-gnueabihf", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-1160a86b-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-1160a86b-v1.zip", - "checksum": "SHA-256:ab71bb61b12ad109750d4d39200c2aeaa826395f65a2c95c2012d969eb5f76a2", - "size": "405894668" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-1160a86b-v2.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-1160a86b-v2.zip", + "checksum": "SHA-256:2050bda30b1d3cc397683f38267744dc45d7e4c4dc857250477117f30db754db", + "size": "406007754" } ] }, From e30e3c30f189aa87d69acb16b672f0ff2a94f1be Mon Sep 17 00:00:00 2001 From: Me No Dev Date: Tue, 25 Feb 2025 10:28:07 +0200 Subject: [PATCH 013/173] IDF master (#10999) * IDF master c71d74e2 * IDF master 81e8b752 * IDF master 877057db --- package/package_esp32_index.template.json | 68 +++++++++++------------ 1 file changed, 34 insertions(+), 34 deletions(-) diff --git a/package/package_esp32_index.template.json b/package/package_esp32_index.template.json index 714328f9a17..46f0cd3418a 100644 --- a/package/package_esp32_index.template.json +++ b/package/package_esp32_index.template.json @@ -51,7 +51,7 @@ { "packager": "esp32", "name": "esp32-arduino-libs", - "version": "idf-master-1160a86b-v2" + "version": "idf-master-877057db-v1" }, { "packager": "esp32", @@ -104,63 +104,63 @@ "tools": [ { "name": "esp32-arduino-libs", - "version": "idf-master-1160a86b-v2", + "version": "idf-master-877057db-v1", "systems": [ { "host": "i686-mingw32", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-1160a86b-v2.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-1160a86b-v2.zip", - "checksum": "SHA-256:2050bda30b1d3cc397683f38267744dc45d7e4c4dc857250477117f30db754db", - "size": "406007754" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-877057db-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-877057db-v1.zip", + "checksum": "SHA-256:fcded8de4107841bea5aaeb7bf7b53a336b3d132df29fb09ae6697b187d51920", + "size": "411930767" }, { "host": "x86_64-mingw32", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-1160a86b-v2.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-1160a86b-v2.zip", - "checksum": "SHA-256:2050bda30b1d3cc397683f38267744dc45d7e4c4dc857250477117f30db754db", - "size": "406007754" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-877057db-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-877057db-v1.zip", + "checksum": "SHA-256:fcded8de4107841bea5aaeb7bf7b53a336b3d132df29fb09ae6697b187d51920", + "size": "411930767" }, { "host": "arm64-apple-darwin", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-1160a86b-v2.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-1160a86b-v2.zip", - "checksum": "SHA-256:2050bda30b1d3cc397683f38267744dc45d7e4c4dc857250477117f30db754db", - "size": "406007754" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-877057db-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-877057db-v1.zip", + "checksum": "SHA-256:fcded8de4107841bea5aaeb7bf7b53a336b3d132df29fb09ae6697b187d51920", + "size": "411930767" }, { "host": "x86_64-apple-darwin", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-1160a86b-v2.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-1160a86b-v2.zip", - "checksum": "SHA-256:2050bda30b1d3cc397683f38267744dc45d7e4c4dc857250477117f30db754db", - "size": "406007754" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-877057db-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-877057db-v1.zip", + "checksum": "SHA-256:fcded8de4107841bea5aaeb7bf7b53a336b3d132df29fb09ae6697b187d51920", + "size": "411930767" }, { "host": "x86_64-pc-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-1160a86b-v2.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-1160a86b-v2.zip", - "checksum": "SHA-256:2050bda30b1d3cc397683f38267744dc45d7e4c4dc857250477117f30db754db", - "size": "406007754" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-877057db-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-877057db-v1.zip", + "checksum": "SHA-256:fcded8de4107841bea5aaeb7bf7b53a336b3d132df29fb09ae6697b187d51920", + "size": "411930767" }, { "host": "i686-pc-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-1160a86b-v2.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-1160a86b-v2.zip", - "checksum": "SHA-256:2050bda30b1d3cc397683f38267744dc45d7e4c4dc857250477117f30db754db", - "size": "406007754" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-877057db-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-877057db-v1.zip", + "checksum": "SHA-256:fcded8de4107841bea5aaeb7bf7b53a336b3d132df29fb09ae6697b187d51920", + "size": "411930767" }, { "host": "aarch64-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-1160a86b-v2.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-1160a86b-v2.zip", - "checksum": "SHA-256:2050bda30b1d3cc397683f38267744dc45d7e4c4dc857250477117f30db754db", - "size": "406007754" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-877057db-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-877057db-v1.zip", + "checksum": "SHA-256:fcded8de4107841bea5aaeb7bf7b53a336b3d132df29fb09ae6697b187d51920", + "size": "411930767" }, { "host": "arm-linux-gnueabihf", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-1160a86b-v2.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-1160a86b-v2.zip", - "checksum": "SHA-256:2050bda30b1d3cc397683f38267744dc45d7e4c4dc857250477117f30db754db", - "size": "406007754" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-877057db-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-877057db-v1.zip", + "checksum": "SHA-256:fcded8de4107841bea5aaeb7bf7b53a336b3d132df29fb09ae6697b187d51920", + "size": "411930767" } ] }, From 3fe2fe5311df6fc93cad8cb19e5a234b37605e3d Mon Sep 17 00:00:00 2001 From: Me No Dev Date: Thu, 27 Feb 2025 00:56:51 +0200 Subject: [PATCH 014/173] IDF master 0461e2ff (#11018) --- package/package_esp32_index.template.json | 68 +++++++++++------------ 1 file changed, 34 insertions(+), 34 deletions(-) diff --git a/package/package_esp32_index.template.json b/package/package_esp32_index.template.json index 46f0cd3418a..f5bf9ed85cd 100644 --- a/package/package_esp32_index.template.json +++ b/package/package_esp32_index.template.json @@ -51,7 +51,7 @@ { "packager": "esp32", "name": "esp32-arduino-libs", - "version": "idf-master-877057db-v1" + "version": "idf-master-0461e2ff-v1" }, { "packager": "esp32", @@ -104,63 +104,63 @@ "tools": [ { "name": "esp32-arduino-libs", - "version": "idf-master-877057db-v1", + "version": "idf-master-0461e2ff-v1", "systems": [ { "host": "i686-mingw32", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-877057db-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-877057db-v1.zip", - "checksum": "SHA-256:fcded8de4107841bea5aaeb7bf7b53a336b3d132df29fb09ae6697b187d51920", - "size": "411930767" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-0461e2ff-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-0461e2ff-v1.zip", + "checksum": "SHA-256:e1413b7d711cbdc4efb3064d6ac25e09068d7a01db3a706bd1426b0fd992e35b", + "size": "411886500" }, { "host": "x86_64-mingw32", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-877057db-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-877057db-v1.zip", - "checksum": "SHA-256:fcded8de4107841bea5aaeb7bf7b53a336b3d132df29fb09ae6697b187d51920", - "size": "411930767" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-0461e2ff-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-0461e2ff-v1.zip", + "checksum": "SHA-256:e1413b7d711cbdc4efb3064d6ac25e09068d7a01db3a706bd1426b0fd992e35b", + "size": "411886500" }, { "host": "arm64-apple-darwin", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-877057db-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-877057db-v1.zip", - "checksum": "SHA-256:fcded8de4107841bea5aaeb7bf7b53a336b3d132df29fb09ae6697b187d51920", - "size": "411930767" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-0461e2ff-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-0461e2ff-v1.zip", + "checksum": "SHA-256:e1413b7d711cbdc4efb3064d6ac25e09068d7a01db3a706bd1426b0fd992e35b", + "size": "411886500" }, { "host": "x86_64-apple-darwin", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-877057db-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-877057db-v1.zip", - "checksum": "SHA-256:fcded8de4107841bea5aaeb7bf7b53a336b3d132df29fb09ae6697b187d51920", - "size": "411930767" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-0461e2ff-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-0461e2ff-v1.zip", + "checksum": "SHA-256:e1413b7d711cbdc4efb3064d6ac25e09068d7a01db3a706bd1426b0fd992e35b", + "size": "411886500" }, { "host": "x86_64-pc-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-877057db-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-877057db-v1.zip", - "checksum": "SHA-256:fcded8de4107841bea5aaeb7bf7b53a336b3d132df29fb09ae6697b187d51920", - "size": "411930767" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-0461e2ff-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-0461e2ff-v1.zip", + "checksum": "SHA-256:e1413b7d711cbdc4efb3064d6ac25e09068d7a01db3a706bd1426b0fd992e35b", + "size": "411886500" }, { "host": "i686-pc-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-877057db-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-877057db-v1.zip", - "checksum": "SHA-256:fcded8de4107841bea5aaeb7bf7b53a336b3d132df29fb09ae6697b187d51920", - "size": "411930767" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-0461e2ff-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-0461e2ff-v1.zip", + "checksum": "SHA-256:e1413b7d711cbdc4efb3064d6ac25e09068d7a01db3a706bd1426b0fd992e35b", + "size": "411886500" }, { "host": "aarch64-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-877057db-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-877057db-v1.zip", - "checksum": "SHA-256:fcded8de4107841bea5aaeb7bf7b53a336b3d132df29fb09ae6697b187d51920", - "size": "411930767" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-0461e2ff-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-0461e2ff-v1.zip", + "checksum": "SHA-256:e1413b7d711cbdc4efb3064d6ac25e09068d7a01db3a706bd1426b0fd992e35b", + "size": "411886500" }, { "host": "arm-linux-gnueabihf", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-877057db-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-877057db-v1.zip", - "checksum": "SHA-256:fcded8de4107841bea5aaeb7bf7b53a336b3d132df29fb09ae6697b187d51920", - "size": "411930767" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-0461e2ff-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-0461e2ff-v1.zip", + "checksum": "SHA-256:e1413b7d711cbdc4efb3064d6ac25e09068d7a01db3a706bd1426b0fd992e35b", + "size": "411886500" } ] }, From d66eeb77547290eb5811fe6662ca2f709e747577 Mon Sep 17 00:00:00 2001 From: Jason2866 <24528715+Jason2866@users.noreply.github.com> Date: Wed, 5 Mar 2025 11:44:51 +0100 Subject: [PATCH 015/173] c5 flash base address is 0x2000 (#11037) * c5 flash base address is 0x2000 * ci(pre-commit): Apply automatic fixes --------- Co-authored-by: pre-commit-ci-lite[bot] <117423508+pre-commit-ci-lite[bot]@users.noreply.github.com> --- cores/esp32/Esp.cpp | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/cores/esp32/Esp.cpp b/cores/esp32/Esp.cpp index 2b2e1280683..f911b2d5f20 100644 --- a/cores/esp32/Esp.cpp +++ b/cores/esp32/Esp.cpp @@ -66,7 +66,7 @@ extern "C" { #define ESP_FLASH_IMAGE_BASE 0x2000 // Esp32p4 is located at 0x2000 #elif CONFIG_IDF_TARGET_ESP32C5 #include "esp32c5/rom/spi_flash.h" -#define ESP_FLASH_IMAGE_BASE 0x0000 // Esp32c5 is located at 0x0000 +#define ESP_FLASH_IMAGE_BASE 0x2000 // Esp32c5 is located at 0x2000 #else #error Target CONFIG_IDF_TARGET is not supported #endif @@ -306,11 +306,11 @@ const char *EspClass::getChipModel(void) { case CHIP_ESP32H2: return "ESP32-H2"; case CHIP_ESP32P4: return "ESP32-P4"; #if ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 5, 0) - case CHIP_ESP32C5: return "ESP32-C5"; + case CHIP_ESP32C5: return "ESP32-C5"; case CHIP_ESP32C61: return "ESP32-C61"; case CHIP_ESP32H21: return "ESP32-H21"; #endif - default: return "UNKNOWN"; + default: return "UNKNOWN"; } #endif } From 7c1ac1ae60672216f942b812dcebc0e97326e9e2 Mon Sep 17 00:00:00 2001 From: Me No Dev Date: Mon, 10 Mar 2025 22:19:40 +0200 Subject: [PATCH 016/173] feat(wifi): Add support for 2.4GHz and 5GHz band switching (#11045) * feat(wifi): Add support for 2.4GHz and 5GHz band switching * ci(pre-commit): Apply automatic fixes --------- Co-authored-by: pre-commit-ci-lite[bot] <117423508+pre-commit-ci-lite[bot]@users.noreply.github.com> --- libraries/WiFi/examples/WiFiScan/WiFiScan.ino | 41 ++++++--- libraries/WiFi/src/WiFiGeneric.cpp | 88 +++++++++++++++++++ libraries/WiFi/src/WiFiGeneric.h | 4 + 3 files changed, 121 insertions(+), 12 deletions(-) diff --git a/libraries/WiFi/examples/WiFiScan/WiFiScan.ino b/libraries/WiFi/examples/WiFiScan/WiFiScan.ino index 15ce367c897..98733adb0bb 100644 --- a/libraries/WiFi/examples/WiFiScan/WiFiScan.ino +++ b/libraries/WiFi/examples/WiFiScan/WiFiScan.ino @@ -1,5 +1,5 @@ /* - * This sketch demonstrates how to scan WiFi networks. + * This sketch demonstrates how to scan WiFi networks. For chips that support 5GHz band, separate scans are done for all bands. * The API is based on the Arduino WiFi Shield library, but has significant changes as newer WiFi functions are supported. * E.g. the return value of `encryptionType()` different because more modern encryption is supported. */ @@ -7,18 +7,13 @@ void setup() { Serial.begin(115200); - - // Set WiFi to station mode and disconnect from an AP if it was previously connected. - WiFi.mode(WIFI_STA); - WiFi.disconnect(); - delay(100); - + // Enable Station Interface + WiFi.STA.begin(); Serial.println("Setup done"); } -void loop() { +void ScanWiFi() { Serial.println("Scan start"); - // WiFi.scanNetworks will return the number of networks found. int n = WiFi.scanNetworks(); Serial.println("Scan done"); @@ -54,11 +49,33 @@ void loop() { delay(10); } } - Serial.println(""); // Delete the scan result to free memory for code below. WiFi.scanDelete(); - + Serial.println("-------------------------------------"); +} +void loop() { + Serial.println("-------------------------------------"); + Serial.println("Default wifi band mode scan:"); + Serial.println("-------------------------------------"); + WiFi.setBandMode(WIFI_BAND_MODE_AUTO); + ScanWiFi(); +#if CONFIG_SOC_WIFI_SUPPORT_5G + // Wait a bit before scanning again. + delay(1000); + Serial.println("-------------------------------------"); + Serial.println("2.4 Ghz wifi band mode scan:"); + Serial.println("-------------------------------------"); + WiFi.setBandMode(WIFI_BAND_MODE_2G_ONLY); + ScanWiFi(); + // Wait a bit before scanning again. + delay(1000); + Serial.println("-------------------------------------"); + Serial.println("5 Ghz wifi band mode scan:"); + Serial.println("-------------------------------------"); + WiFi.setBandMode(WIFI_BAND_MODE_5G_ONLY); + ScanWiFi(); +#endif // Wait a bit before scanning again. - delay(5000); + delay(10000); } diff --git a/libraries/WiFi/src/WiFiGeneric.cpp b/libraries/WiFi/src/WiFiGeneric.cpp index aa994963514..6e38e19a29f 100644 --- a/libraries/WiFi/src/WiFiGeneric.cpp +++ b/libraries/WiFi/src/WiFiGeneric.cpp @@ -378,6 +378,10 @@ static bool espWiFiStart() { log_e("esp_wifi_start 0x%x: %s", err, esp_err_to_name(err)); return _esp_wifi_started; } +#if SOC_WIFI_SUPPORT_5G + log_v("Setting Band Mode to AUTO"); + esp_wifi_set_band_mode(WIFI_BAND_MODE_AUTO); +#endif return _esp_wifi_started; } @@ -729,6 +733,90 @@ wifi_ps_type_t WiFiGenericClass::getSleep() { return _sleepEnabled; } +/** + * control wifi band mode + * @param band_mode enum possible band modes + * @return ok + */ +bool WiFiGenericClass::setBandMode(wifi_band_mode_t band_mode) { +#if SOC_WIFI_SUPPORT_5G + if (!WiFi.STA.started() && !WiFi.AP.started()) { + log_e("You need to start WiFi first"); + return false; + } + wifi_band_mode_t bm = WIFI_BAND_MODE_AUTO; + esp_err_t err = esp_wifi_get_band_mode(&bm); + if (err != ESP_OK) { + log_e("Failed to get Current Band Mode: 0x%x: %s", err, esp_err_to_name(err)); + return false; + } else if (bm == band_mode) { + log_d("No change in Band Mode"); + return true; + } else { + log_d("Switching Band Mode from %d to %d", bm, band_mode); + } +#if ARDUHAL_LOG_LEVEL >= ARDUHAL_LOG_LEVEL_ERROR + if (WiFi.STA.connected() || WiFi.AP.connected()) { + log_e("Your network will get disconnected!"); + } +#endif + err = esp_wifi_set_band_mode(band_mode); + if (err != ESP_OK) { + log_e("Failed to set Band Mode: 0x%x: %s", err, esp_err_to_name(err)); + return false; + } + delay(100); + return true; +#else + if (band_mode == WIFI_BAND_MODE_5G_ONLY) { + log_e("This chip supports only 2.4GHz WiFi"); + } + return band_mode != WIFI_BAND_MODE_5G_ONLY; +#endif +} + +/** + * get the current enabled wifi band mode + * @return enum band mode + */ +wifi_band_mode_t WiFiGenericClass::getBandMode() { +#if SOC_WIFI_SUPPORT_5G + wifi_band_mode_t band_mode = WIFI_BAND_MODE_AUTO; + if (!WiFi.STA.started() && !WiFi.AP.started()) { + log_e("You need to start WiFi first"); + return band_mode; + } + esp_err_t err = esp_wifi_get_band_mode(&band_mode); + if (err != ESP_OK) { + log_e("Failed to get Band Mode: 0x%x: %s", err, esp_err_to_name(err)); + } + return band_mode; +#else + return WIFI_BAND_MODE_2G_ONLY; +#endif +} + +/** + * get the current active wifi band + * @return enum band + */ +wifi_band_t WiFiGenericClass::getBand() { +#if SOC_WIFI_SUPPORT_5G + wifi_band_t band = WIFI_BAND_2G; + if (!WiFi.STA.started() && !WiFi.AP.started()) { + log_e("You need to start WiFi first"); + return band; + } + esp_err_t err = esp_wifi_get_band(&band); + if (err != ESP_OK) { + log_e("Failed to get Band: 0x%x: %s", err, esp_err_to_name(err)); + } + return band; +#else + return WIFI_BAND_2G; +#endif +} + /** * control wifi tx power * @param power enum maximum wifi tx power diff --git a/libraries/WiFi/src/WiFiGeneric.h b/libraries/WiFi/src/WiFiGeneric.h index ed216229ed4..8497b4b4f3d 100644 --- a/libraries/WiFi/src/WiFiGeneric.h +++ b/libraries/WiFi/src/WiFiGeneric.h @@ -111,6 +111,10 @@ class WiFiGenericClass { bool setTxPower(wifi_power_t power); wifi_power_t getTxPower(); + bool setBandMode(wifi_band_mode_t band_mode); + wifi_band_mode_t getBandMode(); + wifi_band_t getBand(); + bool initiateFTM(uint8_t frm_count = 16, uint16_t burst_period = 2, uint8_t channel = 1, const uint8_t *mac = NULL); static bool setDualAntennaConfig(uint8_t gpio_ant1, uint8_t gpio_ant2, wifi_rx_ant_t rx_mode, wifi_tx_ant_t tx_mode); From a61961d5c995106ba89e1361655471938dfc4502 Mon Sep 17 00:00:00 2001 From: Me No Dev Date: Fri, 28 Mar 2025 10:48:39 +0200 Subject: [PATCH 017/173] IDF master (#11150) * IDF master ee77c489 * IDF master ee77c489 * IDF master 50be9735 * IDF master 23c73cdc * IDF master a45d713b --- package/package_esp32_index.template.json | 128 +++++++++++----------- 1 file changed, 64 insertions(+), 64 deletions(-) diff --git a/package/package_esp32_index.template.json b/package/package_esp32_index.template.json index f5bf9ed85cd..1b935fd8acf 100644 --- a/package/package_esp32_index.template.json +++ b/package/package_esp32_index.template.json @@ -51,7 +51,7 @@ { "packager": "esp32", "name": "esp32-arduino-libs", - "version": "idf-master-0461e2ff-v1" + "version": "idf-master-a45d713b-v1" }, { "packager": "esp32", @@ -76,7 +76,7 @@ { "packager": "esp32", "name": "openocd-esp32", - "version": "v0.12.0-esp32-20241016" + "version": "v0.12.0-esp32-20250226" }, { "packager": "esp32", @@ -104,63 +104,63 @@ "tools": [ { "name": "esp32-arduino-libs", - "version": "idf-master-0461e2ff-v1", + "version": "idf-master-a45d713b-v1", "systems": [ { "host": "i686-mingw32", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-0461e2ff-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-0461e2ff-v1.zip", - "checksum": "SHA-256:e1413b7d711cbdc4efb3064d6ac25e09068d7a01db3a706bd1426b0fd992e35b", - "size": "411886500" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-a45d713b-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-a45d713b-v1.zip", + "checksum": "SHA-256:52aec557b9744e721770853c67c4ee6a391debd7bb779f654bea0e5113658786", + "size": "420636315" }, { "host": "x86_64-mingw32", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-0461e2ff-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-0461e2ff-v1.zip", - "checksum": "SHA-256:e1413b7d711cbdc4efb3064d6ac25e09068d7a01db3a706bd1426b0fd992e35b", - "size": "411886500" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-a45d713b-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-a45d713b-v1.zip", + "checksum": "SHA-256:52aec557b9744e721770853c67c4ee6a391debd7bb779f654bea0e5113658786", + "size": "420636315" }, { "host": "arm64-apple-darwin", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-0461e2ff-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-0461e2ff-v1.zip", - "checksum": "SHA-256:e1413b7d711cbdc4efb3064d6ac25e09068d7a01db3a706bd1426b0fd992e35b", - "size": "411886500" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-a45d713b-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-a45d713b-v1.zip", + "checksum": "SHA-256:52aec557b9744e721770853c67c4ee6a391debd7bb779f654bea0e5113658786", + "size": "420636315" }, { "host": "x86_64-apple-darwin", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-0461e2ff-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-0461e2ff-v1.zip", - "checksum": "SHA-256:e1413b7d711cbdc4efb3064d6ac25e09068d7a01db3a706bd1426b0fd992e35b", - "size": "411886500" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-a45d713b-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-a45d713b-v1.zip", + "checksum": "SHA-256:52aec557b9744e721770853c67c4ee6a391debd7bb779f654bea0e5113658786", + "size": "420636315" }, { "host": "x86_64-pc-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-0461e2ff-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-0461e2ff-v1.zip", - "checksum": "SHA-256:e1413b7d711cbdc4efb3064d6ac25e09068d7a01db3a706bd1426b0fd992e35b", - "size": "411886500" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-a45d713b-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-a45d713b-v1.zip", + "checksum": "SHA-256:52aec557b9744e721770853c67c4ee6a391debd7bb779f654bea0e5113658786", + "size": "420636315" }, { "host": "i686-pc-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-0461e2ff-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-0461e2ff-v1.zip", - "checksum": "SHA-256:e1413b7d711cbdc4efb3064d6ac25e09068d7a01db3a706bd1426b0fd992e35b", - "size": "411886500" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-a45d713b-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-a45d713b-v1.zip", + "checksum": "SHA-256:52aec557b9744e721770853c67c4ee6a391debd7bb779f654bea0e5113658786", + "size": "420636315" }, { "host": "aarch64-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-0461e2ff-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-0461e2ff-v1.zip", - "checksum": "SHA-256:e1413b7d711cbdc4efb3064d6ac25e09068d7a01db3a706bd1426b0fd992e35b", - "size": "411886500" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-a45d713b-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-a45d713b-v1.zip", + "checksum": "SHA-256:52aec557b9744e721770853c67c4ee6a391debd7bb779f654bea0e5113658786", + "size": "420636315" }, { "host": "arm-linux-gnueabihf", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-0461e2ff-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-0461e2ff-v1.zip", - "checksum": "SHA-256:e1413b7d711cbdc4efb3064d6ac25e09068d7a01db3a706bd1426b0fd992e35b", - "size": "411886500" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-a45d713b-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-a45d713b-v1.zip", + "checksum": "SHA-256:52aec557b9744e721770853c67c4ee6a391debd7bb779f654bea0e5113658786", + "size": "420636315" } ] }, @@ -414,56 +414,56 @@ }, { "name": "openocd-esp32", - "version": "v0.12.0-esp32-20241016", + "version": "v0.12.0-esp32-20250226", "systems": [ { "host": "x86_64-pc-linux-gnu", - "url": "https://github.com/espressif/openocd-esp32/releases/download/v0.12.0-esp32-20241016/openocd-esp32-linux-amd64-0.12.0-esp32-20241016.tar.gz", - "archiveFileName": "openocd-esp32-linux-amd64-0.12.0-esp32-20241016.tar.gz", - "checksum": "SHA-256:e82b0f036dc99244bead5f09a86e91bb2365cbcd1122ac68261e5647942485df", - "size": "2398717" + "url": "https://github.com/espressif/openocd-esp32/releases/download/v0.12.0-esp32-20250226/openocd-esp32-linux-amd64-0.12.0-esp32-20250226.tar.gz", + "archiveFileName": "openocd-esp32-linux-amd64-0.12.0-esp32-20250226.tar.gz", + "checksum": "SHA-256:914c726342ba5828e53f41aa454f01f317c42d8e6772d3d874593a6960fc4729", + "size": "2414924" }, { "host": "aarch64-linux-gnu", - "url": "https://github.com/espressif/openocd-esp32/releases/download/v0.12.0-esp32-20241016/openocd-esp32-linux-arm64-0.12.0-esp32-20241016.tar.gz", - "archiveFileName": "openocd-esp32-linux-arm64-0.12.0-esp32-20241016.tar.gz", - "checksum": "SHA-256:8f8daf5bd22ec5d2fa9257b0862ec33da18ee677e023fb9a9eb17f74ce208c76", - "size": "2271584" + "url": "https://github.com/espressif/openocd-esp32/releases/download/v0.12.0-esp32-20250226/openocd-esp32-linux-arm64-0.12.0-esp32-20250226.tar.gz", + "archiveFileName": "openocd-esp32-linux-arm64-0.12.0-esp32-20250226.tar.gz", + "checksum": "SHA-256:c44ee99a9209c0234dbbcec86339fd685f5c61a763b29c33eba590bf62db2296", + "size": "2293923" }, { "host": "arm-linux-gnueabihf", - "url": "https://github.com/espressif/openocd-esp32/releases/download/v0.12.0-esp32-20241016/openocd-esp32-linux-armel-0.12.0-esp32-20241016.tar.gz", - "archiveFileName": "openocd-esp32-linux-armel-0.12.0-esp32-20241016.tar.gz", - "checksum": "SHA-256:bc9c020ecf20e2000f76cffa44305fd5bc44d2e688ea78cce423399d33f19767", - "size": "2414206" + "url": "https://github.com/espressif/openocd-esp32/releases/download/v0.12.0-esp32-20250226/openocd-esp32-linux-armel-0.12.0-esp32-20250226.tar.gz", + "archiveFileName": "openocd-esp32-linux-armel-0.12.0-esp32-20250226.tar.gz", + "checksum": "SHA-256:21ab6af3cf05f9290f4d59f1f381d5094dd2755fc528d3d2feb9334348fc0d8d", + "size": "2436071" }, { "host": "x86_64-apple-darwin", - "url": "https://github.com/espressif/openocd-esp32/releases/download/v0.12.0-esp32-20241016/openocd-esp32-macos-0.12.0-esp32-20241016.tar.gz", - "archiveFileName": "openocd-esp32-macos-0.12.0-esp32-20241016.tar.gz", - "checksum": "SHA-256:02a2dffe801a2d005fa9e614d80ff8173395b2cb0b5d3118d0229d094a9946a7", - "size": "2508089" + "url": "https://github.com/espressif/openocd-esp32/releases/download/v0.12.0-esp32-20250226/openocd-esp32-macos-0.12.0-esp32-20250226.tar.gz", + "archiveFileName": "openocd-esp32-macos-0.12.0-esp32-20250226.tar.gz", + "checksum": "SHA-256:0b5751699e93b6d101381611c96216ddff8c7dfd16425c610993fa27993f9a0a", + "size": "2525387" }, { "host": "arm64-apple-darwin", - "url": "https://github.com/espressif/openocd-esp32/releases/download/v0.12.0-esp32-20241016/openocd-esp32-macos-arm64-0.12.0-esp32-20241016.tar.gz", - "archiveFileName": "openocd-esp32-macos-arm64-0.12.0-esp32-20241016.tar.gz", - "checksum": "SHA-256:c382f9e884d6565cb6089bff5f200f4810994667d885f062c3d3c5625a0fa9d6", - "size": "2552569" + "url": "https://github.com/espressif/openocd-esp32/releases/download/v0.12.0-esp32-20250226/openocd-esp32-macos-arm64-0.12.0-esp32-20250226.tar.gz", + "archiveFileName": "openocd-esp32-macos-arm64-0.12.0-esp32-20250226.tar.gz", + "checksum": "SHA-256:8bffbbb594b27a4971a3922792135f8c836fff26991f7f450094386920263531", + "size": "2568843" }, { "host": "i686-mingw32", - "url": "https://github.com/espressif/openocd-esp32/releases/download/v0.12.0-esp32-20241016/openocd-esp32-win32-0.12.0-esp32-20241016.zip", - "archiveFileName": "openocd-esp32-win32-0.12.0-esp32-20241016.zip", - "checksum": "SHA-256:3b5d615e0a72cc771a45dd469031312d5881c01d7b6bc9edb29b8b6bda8c2e90", - "size": "2946244" + "url": "https://github.com/espressif/openocd-esp32/releases/download/v0.12.0-esp32-20250226/openocd-esp32-win32-0.12.0-esp32-20250226.zip", + "archiveFileName": "openocd-esp32-win32-0.12.0-esp32-20250226.zip", + "checksum": "SHA-256:aaf3c955bb4eb47805a1ba108dfd07a8a56ce720cb40194a354362b5f0961230", + "size": "2960226" }, { "host": "x86_64-mingw32", - "url": "https://github.com/espressif/openocd-esp32/releases/download/v0.12.0-esp32-20241016/openocd-esp32-win64-0.12.0-esp32-20241016.zip", - "archiveFileName": "openocd-esp32-win64-0.12.0-esp32-20241016.zip", - "checksum": "SHA-256:5e7b2fd1947d3a8625f6a11db7a2340cf2f41ff4c61284c022c7d7c32b18780a", - "size": "2946244" + "url": "https://github.com/espressif/openocd-esp32/releases/download/v0.12.0-esp32-20250226/openocd-esp32-win64-0.12.0-esp32-20250226.zip", + "archiveFileName": "openocd-esp32-win64-0.12.0-esp32-20250226.zip", + "checksum": "SHA-256:79baf35325117a53093b62f6b9bee677dd12275d7066e3f8a274d2a80e986b6e", + "size": "2960225" } ] }, From 42ae2426affacf7162770b9399b486eb7954203f Mon Sep 17 00:00:00 2001 From: Jason2866 <24528715+Jason2866@users.noreply.github.com> Date: Mon, 21 Apr 2025 18:32:51 +0200 Subject: [PATCH 018/173] fix C5 compile (#11255) --- cores/esp32/esp32-hal-uart.c | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/cores/esp32/esp32-hal-uart.c b/cores/esp32/esp32-hal-uart.c index ca88edbca27..21540b3e95a 100644 --- a/cores/esp32/esp32-hal-uart.c +++ b/cores/esp32/esp32-hal-uart.c @@ -1157,7 +1157,11 @@ bool uartSetClockSource(uint8_t uartNum, uart_sclk_t clkSrc) { if (uart->num >= SOC_UART_HP_NUM) { switch (clkSrc) { case UART_SCLK_XTAL: uart->_uart_clock_source = LP_UART_SCLK_XTAL_D2; break; +#if CONFIG_IDF_TARGET_ESP32C5 + case UART_SCLK_RTC: uart->_uart_clock_source = LP_UART_SCLK_RC_FAST; break; +#else case UART_SCLK_RTC: uart->_uart_clock_source = LP_UART_SCLK_LP_FAST; break; +#endif case UART_SCLK_DEFAULT: default: uart->_uart_clock_source = LP_UART_SCLK_DEFAULT; } From 03e9c45084c389436869f1ae649549e245866ade Mon Sep 17 00:00:00 2001 From: Me No Dev Date: Wed, 23 Apr 2025 15:50:58 +0300 Subject: [PATCH 019/173] IDF master (#11204) * fix(esp-now): Update TX Callback * IDF master d930a386 --------- Co-authored-by: Jason2866 <24528715+Jason2866@users.noreply.github.com> --- libraries/ESP_NOW/src/ESP32_NOW.cpp | 5 + package/package_esp32_index.template.json | 204 +++++++++++----------- 2 files changed, 107 insertions(+), 102 deletions(-) diff --git a/libraries/ESP_NOW/src/ESP32_NOW.cpp b/libraries/ESP_NOW/src/ESP32_NOW.cpp index 6fd3ff0a0b1..83fec4c4529 100644 --- a/libraries/ESP_NOW/src/ESP32_NOW.cpp +++ b/libraries/ESP_NOW/src/ESP32_NOW.cpp @@ -129,7 +129,12 @@ static void _esp_now_rx_cb(const esp_now_recv_info_t *info, const uint8_t *data, } } +#if ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 5, 0) +static void _esp_now_tx_cb(const esp_now_send_info_t *tx_info, esp_now_send_status_t status) { + const uint8_t *mac_addr = tx_info->des_addr; +#else static void _esp_now_tx_cb(const uint8_t *mac_addr, esp_now_send_status_t status) { +#endif log_v(MACSTR " : %s", MAC2STR(mac_addr), (status == ESP_NOW_SEND_SUCCESS) ? "SUCCESS" : "FAILED"); //find the peer and call it's callback for (uint8_t i = 0; i < ESP_NOW_MAX_TOTAL_PEER_NUM; i++) { diff --git a/package/package_esp32_index.template.json b/package/package_esp32_index.template.json index 1b935fd8acf..a0d78ebc47e 100644 --- a/package/package_esp32_index.template.json +++ b/package/package_esp32_index.template.json @@ -51,7 +51,7 @@ { "packager": "esp32", "name": "esp32-arduino-libs", - "version": "idf-master-a45d713b-v1" + "version": "idf-master-d930a386-v1" }, { "packager": "esp32", @@ -61,7 +61,7 @@ { "packager": "esp32", "name": "xtensa-esp-elf-gdb", - "version": "15.2_20241112" + "version": "16.2_20250324" }, { "packager": "esp32", @@ -71,7 +71,7 @@ { "packager": "esp32", "name": "riscv32-esp-elf-gdb", - "version": "15.2_20241112" + "version": "16.2_20250324" }, { "packager": "esp32", @@ -104,63 +104,63 @@ "tools": [ { "name": "esp32-arduino-libs", - "version": "idf-master-a45d713b-v1", + "version": "idf-master-d930a386-v1", "systems": [ { "host": "i686-mingw32", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-a45d713b-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-a45d713b-v1.zip", - "checksum": "SHA-256:52aec557b9744e721770853c67c4ee6a391debd7bb779f654bea0e5113658786", - "size": "420636315" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-d930a386-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-d930a386-v1.zip", + "checksum": "SHA-256:0310daa4f08f807f2bf3babd2587c2694df64c70e367863eadf5020636b717ae", + "size": "422376381" }, { "host": "x86_64-mingw32", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-a45d713b-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-a45d713b-v1.zip", - "checksum": "SHA-256:52aec557b9744e721770853c67c4ee6a391debd7bb779f654bea0e5113658786", - "size": "420636315" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-d930a386-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-d930a386-v1.zip", + "checksum": "SHA-256:0310daa4f08f807f2bf3babd2587c2694df64c70e367863eadf5020636b717ae", + "size": "422376381" }, { "host": "arm64-apple-darwin", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-a45d713b-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-a45d713b-v1.zip", - "checksum": "SHA-256:52aec557b9744e721770853c67c4ee6a391debd7bb779f654bea0e5113658786", - "size": "420636315" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-d930a386-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-d930a386-v1.zip", + "checksum": "SHA-256:0310daa4f08f807f2bf3babd2587c2694df64c70e367863eadf5020636b717ae", + "size": "422376381" }, { "host": "x86_64-apple-darwin", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-a45d713b-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-a45d713b-v1.zip", - "checksum": "SHA-256:52aec557b9744e721770853c67c4ee6a391debd7bb779f654bea0e5113658786", - "size": "420636315" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-d930a386-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-d930a386-v1.zip", + "checksum": "SHA-256:0310daa4f08f807f2bf3babd2587c2694df64c70e367863eadf5020636b717ae", + "size": "422376381" }, { "host": "x86_64-pc-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-a45d713b-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-a45d713b-v1.zip", - "checksum": "SHA-256:52aec557b9744e721770853c67c4ee6a391debd7bb779f654bea0e5113658786", - "size": "420636315" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-d930a386-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-d930a386-v1.zip", + "checksum": "SHA-256:0310daa4f08f807f2bf3babd2587c2694df64c70e367863eadf5020636b717ae", + "size": "422376381" }, { "host": "i686-pc-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-a45d713b-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-a45d713b-v1.zip", - "checksum": "SHA-256:52aec557b9744e721770853c67c4ee6a391debd7bb779f654bea0e5113658786", - "size": "420636315" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-d930a386-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-d930a386-v1.zip", + "checksum": "SHA-256:0310daa4f08f807f2bf3babd2587c2694df64c70e367863eadf5020636b717ae", + "size": "422376381" }, { "host": "aarch64-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-a45d713b-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-a45d713b-v1.zip", - "checksum": "SHA-256:52aec557b9744e721770853c67c4ee6a391debd7bb779f654bea0e5113658786", - "size": "420636315" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-d930a386-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-d930a386-v1.zip", + "checksum": "SHA-256:0310daa4f08f807f2bf3babd2587c2694df64c70e367863eadf5020636b717ae", + "size": "422376381" }, { "host": "arm-linux-gnueabihf", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-a45d713b-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-a45d713b-v1.zip", - "checksum": "SHA-256:52aec557b9744e721770853c67c4ee6a391debd7bb779f654bea0e5113658786", - "size": "420636315" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-d930a386-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-d930a386-v1.zip", + "checksum": "SHA-256:0310daa4f08f807f2bf3babd2587c2694df64c70e367863eadf5020636b717ae", + "size": "422376381" } ] }, @@ -228,63 +228,63 @@ }, { "name": "xtensa-esp-elf-gdb", - "version": "15.2_20241112", + "version": "16.2_20250324", "systems": [ { "host": "x86_64-pc-linux-gnu", - "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v15.2_20241112/xtensa-esp-elf-gdb-15.2_20241112-x86_64-linux-gnu.tar.gz", - "archiveFileName": "xtensa-esp-elf-gdb-15.2_20241112-x86_64-linux-gnu.tar.gz", - "checksum": "SHA-256:18774349d2b1c7d7f5ba984563f7022aef4a3df4b706ed8821c53266f599343d", - "size": "35179121" + "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v16.2_20250324/xtensa-esp-elf-gdb-16.2_20250324-x86_64-linux-gnu.tar.gz", + "archiveFileName": "xtensa-esp-elf-gdb-16.2_20250324-x86_64-linux-gnu.tar.gz", + "checksum": "SHA-256:27b58ab12248e04277c4fdc74038cf0a001d5142df091ab94939ad35053738fd", + "size": "36361058" }, { "host": "aarch64-linux-gnu", - "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v15.2_20241112/xtensa-esp-elf-gdb-15.2_20241112-aarch64-linux-gnu.tar.gz", - "archiveFileName": "xtensa-esp-elf-gdb-15.2_20241112-aarch64-linux-gnu.tar.gz", - "checksum": "SHA-256:77cb3b2c85d6cfbb40b7f99eebbc2b1c3f4fe13eba20b3da798bdbbc6eb8e87c", - "size": "34295046" + "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v16.2_20250324/xtensa-esp-elf-gdb-16.2_20250324-aarch64-linux-gnu.tar.gz", + "archiveFileName": "xtensa-esp-elf-gdb-16.2_20250324-aarch64-linux-gnu.tar.gz", + "checksum": "SHA-256:24f85aa778e1605098a13ff7bd29d5760767faf012705c8915cb08b32cad0206", + "size": "35442104" }, { "host": "arm-linux-gnueabihf", - "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v15.2_20241112/xtensa-esp-elf-gdb-15.2_20241112-arm-linux-gnueabi.tar.gz", - "archiveFileName": "xtensa-esp-elf-gdb-15.2_20241112-arm-linux-gnueabi.tar.gz", - "checksum": "SHA-256:b87af0539de118eb9d43a4a89c8c1b0a6ab2557560015155104c44f91b5c4aa0", - "size": "30338727" + "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v16.2_20250324/xtensa-esp-elf-gdb-16.2_20250324-arm-linux-gnueabi.tar.gz", + "archiveFileName": "xtensa-esp-elf-gdb-16.2_20250324-arm-linux-gnueabi.tar.gz", + "checksum": "SHA-256:c73e43038b6d50374cd0ee714370ce748189e0b00404d581babd2bb0115c4785", + "size": "31260410" }, { "host": "i686-pc-linux-gnu", - "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v15.2_20241112/xtensa-esp-elf-gdb-15.2_20241112-i586-linux-gnu.tar.gz", - "archiveFileName": "xtensa-esp-elf-gdb-15.2_20241112-i586-linux-gnu.tar.gz", - "checksum": "SHA-256:2c7531bd390928fed479999ac9089b39a25f56ca4f4cc330db7d7a633a37405b", - "size": "33906121" + "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v16.2_20250324/xtensa-esp-elf-gdb-16.2_20250324-i586-linux-gnu.tar.gz", + "archiveFileName": "xtensa-esp-elf-gdb-16.2_20250324-i586-linux-gnu.tar.gz", + "checksum": "SHA-256:dc7b8aad0fb1c6a1abfdb8dff4f08221ea08a0f28fb837f181969ac1174d4dc6", + "size": "35067894" }, { "host": "x86_64-apple-darwin", - "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v15.2_20241112/xtensa-esp-elf-gdb-15.2_20241112-x86_64-apple-darwin21.1.tar.gz", - "archiveFileName": "xtensa-esp-elf-gdb-15.2_20241112-x86_64-apple-darwin21.1.tar.gz", - "checksum": "SHA-256:ba2907be9a4c22c4e418f42ec84cf57401570a71dabbe69425227114ebf351a7", - "size": "52502302" + "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v16.2_20250324/xtensa-esp-elf-gdb-16.2_20250324-x86_64-apple-darwin21.1.tar.gz", + "archiveFileName": "xtensa-esp-elf-gdb-16.2_20250324-x86_64-apple-darwin21.1.tar.gz", + "checksum": "SHA-256:398c429cfe696bad01d636c5488cadc87b20471c1b5ed02c60eee5ef2a775c93", + "size": "54992785" }, { "host": "arm64-apple-darwin", - "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v15.2_20241112/xtensa-esp-elf-gdb-15.2_20241112-aarch64-apple-darwin21.1.tar.gz", - "archiveFileName": "xtensa-esp-elf-gdb-15.2_20241112-aarch64-apple-darwin21.1.tar.gz", - "checksum": "SHA-256:893500d6de354a6870820b9398531d8cd37d1cd85f3ae9b1f8a4c070b8048707", - "size": "41892363" + "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v16.2_20250324/xtensa-esp-elf-gdb-16.2_20250324-aarch64-apple-darwin21.1.tar.gz", + "archiveFileName": "xtensa-esp-elf-gdb-16.2_20250324-aarch64-apple-darwin21.1.tar.gz", + "checksum": "SHA-256:b6d85c0d76d653bb55f9d06b0cd509eab7e99db541c88b8c849c64827e9d74a9", + "size": "43538967" }, { "host": "i686-mingw32", - "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v15.2_20241112/xtensa-esp-elf-gdb-15.2_20241112-i686-w64-mingw32.zip", - "archiveFileName": "xtensa-esp-elf-gdb-15.2_20241112-i686-w64-mingw32.zip", - "checksum": "SHA-256:328181380fccb252105c51a86071edbc5ef63e0114540edc4f8b2d62acf44916", - "size": "31307770" + "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v16.2_20250324/xtensa-esp-elf-gdb-16.2_20250324-i686-w64-mingw32.zip", + "archiveFileName": "xtensa-esp-elf-gdb-16.2_20250324-i686-w64-mingw32.zip", + "checksum": "SHA-256:f748d6b65fdf66733b82e12d0d85a05e3134122416280379df129cfebe2aa4b2", + "size": "32189419" }, { "host": "x86_64-mingw32", - "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v15.2_20241112/xtensa-esp-elf-gdb-15.2_20241112-x86_64-w64-mingw32.zip", - "archiveFileName": "xtensa-esp-elf-gdb-15.2_20241112-x86_64-w64-mingw32.zip", - "checksum": "SHA-256:9c1949058d7aa1fa6f6f2d03173659b9f54e3c3940cbeebc1d56ae169c604ab2", - "size": "31420687" + "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v16.2_20250324/xtensa-esp-elf-gdb-16.2_20250324-x86_64-w64-mingw32.zip", + "archiveFileName": "xtensa-esp-elf-gdb-16.2_20250324-x86_64-w64-mingw32.zip", + "checksum": "SHA-256:e970fc3ec8a1d0acee2432e91e0a01b348613a0425aacfa981b2fc505fe920cc", + "size": "32290997" } ] }, @@ -352,63 +352,63 @@ }, { "name": "riscv32-esp-elf-gdb", - "version": "15.2_20241112", + "version": "16.2_20250324", "systems": [ { "host": "x86_64-pc-linux-gnu", - "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v15.2_20241112/riscv32-esp-elf-gdb-15.2_20241112-x86_64-linux-gnu.tar.gz", - "archiveFileName": "riscv32-esp-elf-gdb-15.2_20241112-x86_64-linux-gnu.tar.gz", - "checksum": "SHA-256:bfca245b3d84244ad3b6156496728d916ac5ccc0d7f8e048194b7eba5cdbe047", - "size": "35297398" + "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v16.2_20250324/riscv32-esp-elf-gdb-16.2_20250324-x86_64-linux-gnu.tar.gz", + "archiveFileName": "riscv32-esp-elf-gdb-16.2_20250324-x86_64-linux-gnu.tar.gz", + "checksum": "SHA-256:f9b172d8d72d0a1e2b0b80127df29263a0cb0d0c4e998e09c27031bfac09f3ec", + "size": "36528201" }, { "host": "aarch64-linux-gnu", - "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v15.2_20241112/riscv32-esp-elf-gdb-15.2_20241112-aarch64-linux-gnu.tar.gz", - "archiveFileName": "riscv32-esp-elf-gdb-15.2_20241112-aarch64-linux-gnu.tar.gz", - "checksum": "SHA-256:8536da9e3093b8f25e0b5204b04ed4afea432d1fd262f2abb466016d3d750ea6", - "size": "34455317" + "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v16.2_20250324/riscv32-esp-elf-gdb-16.2_20250324-aarch64-linux-gnu.tar.gz", + "archiveFileName": "riscv32-esp-elf-gdb-16.2_20250324-aarch64-linux-gnu.tar.gz", + "checksum": "SHA-256:68bb6a85fb58b8a738f799e8fb4fa1f56cfeffc4de803ceb03c8a33cb2cd919d", + "size": "35643464" }, { "host": "arm-linux-gnueabihf", - "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v15.2_20241112/riscv32-esp-elf-gdb-15.2_20241112-arm-linux-gnueabi.tar.gz", - "archiveFileName": "riscv32-esp-elf-gdb-15.2_20241112-arm-linux-gnueabi.tar.gz", - "checksum": "SHA-256:33a80d5e6604bb7d08b15492b8ec84c9176245e066d0bf8aad7570786ca44081", - "size": "31188203" + "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v16.2_20250324/riscv32-esp-elf-gdb-16.2_20250324-arm-linux-gnueabi.tar.gz", + "archiveFileName": "riscv32-esp-elf-gdb-16.2_20250324-arm-linux-gnueabi.tar.gz", + "checksum": "SHA-256:673038ab9fb2b7391ff9252824194e3b9e40668efe9ce54d1e582a9d6c51f04a", + "size": "32154574" }, { "host": "i686-pc-linux-gnu", - "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v15.2_20241112/riscv32-esp-elf-gdb-15.2_20241112-i586-linux-gnu.tar.gz", - "archiveFileName": "riscv32-esp-elf-gdb-15.2_20241112-i586-linux-gnu.tar.gz", - "checksum": "SHA-256:ba448ecf2c80064013eaacb0e7be514a5ef17516f49ff3e50bc41c5578b8d88e", - "size": "34211289" + "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v16.2_20250324/riscv32-esp-elf-gdb-16.2_20250324-i586-linux-gnu.tar.gz", + "archiveFileName": "riscv32-esp-elf-gdb-16.2_20250324-i586-linux-gnu.tar.gz", + "checksum": "SHA-256:62f05d5fe08145b25e423dd0b3f1ae260be99abf5462b8cfd918bf2231e26e30", + "size": "35410891" }, { "host": "x86_64-apple-darwin", - "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v15.2_20241112/riscv32-esp-elf-gdb-15.2_20241112-x86_64-apple-darwin21.1.tar.gz", - "archiveFileName": "riscv32-esp-elf-gdb-15.2_20241112-x86_64-apple-darwin21.1.tar.gz", - "checksum": "SHA-256:bdd07c54fe3216eb5c34f92cac514eb4af777c951573f186c33408c75f56bb4b", - "size": "52831957" + "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v16.2_20250324/riscv32-esp-elf-gdb-16.2_20250324-x86_64-apple-darwin21.1.tar.gz", + "archiveFileName": "riscv32-esp-elf-gdb-16.2_20250324-x86_64-apple-darwin21.1.tar.gz", + "checksum": "SHA-256:63ae12cfbab648e2d2ca7a700a0c615c4f36a6fbe6876c11ba108115ee0d60f2", + "size": "55359246" }, { "host": "arm64-apple-darwin", - "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v15.2_20241112/riscv32-esp-elf-gdb-15.2_20241112-aarch64-apple-darwin21.1.tar.gz", - "archiveFileName": "riscv32-esp-elf-gdb-15.2_20241112-aarch64-apple-darwin21.1.tar.gz", - "checksum": "SHA-256:9924f439ae77c3346e532a48a0d56330466c33ce1abd8d72b77f9f7f5c3b1196", - "size": "42227631" + "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v16.2_20250324/riscv32-esp-elf-gdb-16.2_20250324-aarch64-apple-darwin21.1.tar.gz", + "archiveFileName": "riscv32-esp-elf-gdb-16.2_20250324-aarch64-apple-darwin21.1.tar.gz", + "checksum": "SHA-256:bfbe49774f839020cef988537da0a06896dfe4a382674c62285361ed9bd4aee3", + "size": "43926592" }, { "host": "i686-mingw32", - "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v15.2_20241112/riscv32-esp-elf-gdb-15.2_20241112-i686-w64-mingw32.zip", - "archiveFileName": "riscv32-esp-elf-gdb-15.2_20241112-i686-w64-mingw32.zip", - "checksum": "SHA-256:dcdafd30854b092671f555d844d94b0733e2ffcac8c026b4fb7b8a72ebef0016", - "size": "31971712" + "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v16.2_20250324/riscv32-esp-elf-gdb-16.2_20250324-i686-w64-mingw32.zip", + "archiveFileName": "riscv32-esp-elf-gdb-16.2_20250324-i686-w64-mingw32.zip", + "checksum": "SHA-256:e8b84eec990ff514729b3770edf2b543f36670f43663ce0c3b624fb4884812ca", + "size": "32914955" }, { "host": "x86_64-mingw32", - "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v15.2_20241112/riscv32-esp-elf-gdb-15.2_20241112-x86_64-w64-mingw32.zip", - "archiveFileName": "riscv32-esp-elf-gdb-15.2_20241112-x86_64-w64-mingw32.zip", - "checksum": "SHA-256:6ec8b3a073f2c5835321b4b560f8fc56180458204d071da139ff983436ea45e5", - "size": "31759225" + "url": "https://github.com/espressif/binutils-gdb/releases/download/esp-gdb-v16.2_20250324/riscv32-esp-elf-gdb-16.2_20250324-x86_64-w64-mingw32.zip", + "archiveFileName": "riscv32-esp-elf-gdb-16.2_20250324-x86_64-w64-mingw32.zip", + "checksum": "SHA-256:37c79178900c19ca7487c26af4b5ad6b0d3f34683bd0e9c2ddd39038c999e429", + "size": "32667353" } ] }, From b115acea40b2381bc00e07944d926fe726d05ef5 Mon Sep 17 00:00:00 2001 From: Me No Dev Date: Thu, 1 May 2025 02:15:03 +0300 Subject: [PATCH 020/173] IDF master (#11289) * fix(libs): Ensure compilation with ESP32-C5 * fix(i2c): Update I2C Slave init call * IDF master 465b159c * ci(simple_ble): Add check for BLE supported * IDF master 38628f98 --------- Co-authored-by: Lucas Saavedra Vaz <32426024+lucasssvaz@users.noreply.github.com> --- cores/esp32/esp32-hal-i2c-slave.c | 7 ++ idf_component.yml | 18 ++--- libraries/BluetoothSerial/src/BTAddress.cpp | 4 +- libraries/BluetoothSerial/src/BTAddress.h | 4 +- .../BluetoothSerial/src/BTAdvertisedDevice.h | 6 +- .../src/BTAdvertisedDeviceSet.cpp | 4 +- libraries/BluetoothSerial/src/BTScan.h | 7 +- .../BluetoothSerial/src/BTScanResultsSet.cpp | 4 +- .../BluetoothSerial/src/BluetoothSerial.cpp | 3 +- .../BluetoothSerial/src/BluetoothSerial.h | 3 +- .../examples/SimpleBleDevice/ci.json | 1 + libraries/SimpleBLE/src/SimpleBLE.cpp | 3 +- libraries/SimpleBLE/src/SimpleBLE.h | 3 +- package/package_esp32_index.template.json | 68 +++++++++---------- 14 files changed, 80 insertions(+), 55 deletions(-) diff --git a/cores/esp32/esp32-hal-i2c-slave.c b/cores/esp32/esp32-hal-i2c-slave.c index 0e01259da61..1d92a55ae15 100644 --- a/cores/esp32/esp32-hal-i2c-slave.c +++ b/cores/esp32/esp32-hal-i2c-slave.c @@ -43,6 +43,7 @@ #include "soc/i2c_struct.h" #include "soc/periph_defs.h" #include "hal/i2c_ll.h" +#include "hal/i2c_types.h" #ifndef CONFIG_IDF_TARGET_ESP32C5 #include "hal/clk_gate_ll.h" #endif @@ -337,7 +338,13 @@ esp_err_t i2cSlaveInit(uint8_t num, int sda, int scl, uint16_t slaveID, uint32_t } #endif // !defined(CONFIG_IDF_TARGET_ESP32P4) +#if ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 5, 0) + i2c_ll_set_mode(i2c->dev, I2C_BUS_MODE_SLAVE); + i2c_ll_enable_pins_open_drain(i2c->dev, true); +#else i2c_ll_slave_init(i2c->dev); +#endif + #if ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 4, 0) i2c_ll_enable_fifo_mode(i2c->dev, true); #else diff --git a/idf_component.yml b/idf_component.yml index c1fc614cae5..f0e3f84b28d 100644 --- a/idf_component.yml +++ b/idf_component.yml @@ -57,12 +57,12 @@ dependencies: version: "==1.6.3" require: public rules: - - if: "target not in [esp32c2, esp32p4]" + - if: "target not in [esp32c2, esp32p4, esp32c5]" espressif/esp-zigbee-lib: version: "==1.6.3" require: public rules: - - if: "target not in [esp32c2, esp32p4]" + - if: "target not in [esp32c2, esp32p4, esp32c5]" espressif/esp-dsp: version: "^1.3.4" rules: @@ -73,32 +73,32 @@ dependencies: espressif/esp_rainmaker: version: "1.5.2" rules: - - if: "target not in [esp32c2, esp32p4]" + - if: "target not in [esp32c2, esp32p4, esp32c5]" espressif/rmaker_common: version: "1.4.6" rules: - - if: "target not in [esp32c2, esp32p4]" + - if: "target not in [esp32c2, esp32p4, esp32c5]" espressif/esp_insights: version: "1.2.2" rules: - - if: "target not in [esp32c2, esp32p4]" + - if: "target not in [esp32c2, esp32p4, esp32c5]" # New version breaks esp_insights 1.0.1 espressif/esp_diag_data_store: version: "1.0.2" rules: - - if: "target not in [esp32c2, esp32p4]" + - if: "target not in [esp32c2, esp32p4, esp32c5]" espressif/esp_diagnostics: version: "1.2.1" rules: - - if: "target not in [esp32c2, esp32p4]" + - if: "target not in [esp32c2, esp32p4, esp32c5]" espressif/cbor: version: "0.6.0~1" rules: - - if: "target not in [esp32c2, esp32p4]" + - if: "target not in [esp32c2, esp32p4, esp32c5]" espressif/qrcode: version: "0.1.0~2" rules: - - if: "target not in [esp32c2, esp32p4]" + - if: "target not in [esp32c2, esp32p4, esp32c5]" # RainMaker End espressif/esp-sr: version: "^1.4.2" diff --git a/libraries/BluetoothSerial/src/BTAddress.cpp b/libraries/BluetoothSerial/src/BTAddress.cpp index 6a6de6522bd..6dc05f3aafb 100644 --- a/libraries/BluetoothSerial/src/BTAddress.cpp +++ b/libraries/BluetoothSerial/src/BTAddress.cpp @@ -7,7 +7,9 @@ * Author: Thomas M. (ArcticSnowSky) */ #include "sdkconfig.h" -#if defined(CONFIG_BT_ENABLED) && defined(CONFIG_BLUEDROID_ENABLED) +#include "soc/soc_caps.h" + +#if SOC_BT_SUPPORTED && defined(CONFIG_BT_ENABLED) && defined(CONFIG_BLUEDROID_ENABLED) #include "BTAddress.h" #include diff --git a/libraries/BluetoothSerial/src/BTAddress.h b/libraries/BluetoothSerial/src/BTAddress.h index a2af9247fb0..ece3ae83525 100644 --- a/libraries/BluetoothSerial/src/BTAddress.h +++ b/libraries/BluetoothSerial/src/BTAddress.h @@ -10,7 +10,9 @@ #ifndef COMPONENTS_CPP_UTILS_BTADDRESS_H_ #define COMPONENTS_CPP_UTILS_BTADDRESS_H_ #include "sdkconfig.h" -#if defined(CONFIG_BT_ENABLED) && defined(CONFIG_BLUEDROID_ENABLED) +#include "soc/soc_caps.h" + +#if SOC_BT_SUPPORTED && defined(CONFIG_BT_ENABLED) && defined(CONFIG_BLUEDROID_ENABLED) #include // ESP32 BT #include diff --git a/libraries/BluetoothSerial/src/BTAdvertisedDevice.h b/libraries/BluetoothSerial/src/BTAdvertisedDevice.h index 63c19c908b6..53aa2629b56 100644 --- a/libraries/BluetoothSerial/src/BTAdvertisedDevice.h +++ b/libraries/BluetoothSerial/src/BTAdvertisedDevice.h @@ -5,9 +5,11 @@ * Author: Thomas M. (ArcticSnowSky) */ -#ifndef __BTADVERTISEDDEVICE_H__ -#define __BTADVERTISEDDEVICE_H__ +#pragma once +#include "sdkconfig.h" +#include "soc/soc_caps.h" +#if SOC_BT_SUPPORTED && defined(CONFIG_BT_ENABLED) && defined(CONFIG_BLUEDROID_ENABLED) #include "BTAddress.h" #include diff --git a/libraries/BluetoothSerial/src/BTAdvertisedDeviceSet.cpp b/libraries/BluetoothSerial/src/BTAdvertisedDeviceSet.cpp index ed6076a3103..9afc28547e5 100644 --- a/libraries/BluetoothSerial/src/BTAdvertisedDeviceSet.cpp +++ b/libraries/BluetoothSerial/src/BTAdvertisedDeviceSet.cpp @@ -6,7 +6,9 @@ */ #include "sdkconfig.h" -#if defined(CONFIG_BT_ENABLED) && defined(CONFIG_BLUEDROID_ENABLED) +#include "soc/soc_caps.h" + +#if SOC_BT_SUPPORTED && defined(CONFIG_BT_ENABLED) && defined(CONFIG_BLUEDROID_ENABLED) //#include diff --git a/libraries/BluetoothSerial/src/BTScan.h b/libraries/BluetoothSerial/src/BTScan.h index a08f68cd7c2..6fd2daf9f6d 100644 --- a/libraries/BluetoothSerial/src/BTScan.h +++ b/libraries/BluetoothSerial/src/BTScan.h @@ -5,8 +5,11 @@ * Author: Thomas M. (ArcticSnowSky) */ -#ifndef __BTSCAN_H__ -#define __BTSCAN_H__ +#pragma once +#include "sdkconfig.h" +#include "soc/soc_caps.h" + +#if SOC_BT_SUPPORTED && defined(CONFIG_BT_ENABLED) && defined(CONFIG_BLUEDROID_ENABLED) #include #include diff --git a/libraries/BluetoothSerial/src/BTScanResultsSet.cpp b/libraries/BluetoothSerial/src/BTScanResultsSet.cpp index 3633c010eae..02459b081ba 100644 --- a/libraries/BluetoothSerial/src/BTScanResultsSet.cpp +++ b/libraries/BluetoothSerial/src/BTScanResultsSet.cpp @@ -6,7 +6,9 @@ */ #include "sdkconfig.h" -#if defined(CONFIG_BT_ENABLED) && defined(CONFIG_BLUEDROID_ENABLED) +#include "soc/soc_caps.h" + +#if SOC_BT_SUPPORTED && defined(CONFIG_BT_ENABLED) && defined(CONFIG_BLUEDROID_ENABLED) #include diff --git a/libraries/BluetoothSerial/src/BluetoothSerial.cpp b/libraries/BluetoothSerial/src/BluetoothSerial.cpp index 3d00504c1b1..b7eede93ee6 100644 --- a/libraries/BluetoothSerial/src/BluetoothSerial.cpp +++ b/libraries/BluetoothSerial/src/BluetoothSerial.cpp @@ -19,8 +19,9 @@ #include #include "freertos/FreeRTOS.h" #include "freertos/task.h" +#include "soc/soc_caps.h" -#if defined(CONFIG_BT_ENABLED) && defined(CONFIG_BLUEDROID_ENABLED) +#if SOC_BT_SUPPORTED && defined(CONFIG_BT_ENABLED) && defined(CONFIG_BLUEDROID_ENABLED) #ifdef ARDUINO_ARCH_ESP32 #include "esp32-hal-log.h" diff --git a/libraries/BluetoothSerial/src/BluetoothSerial.h b/libraries/BluetoothSerial/src/BluetoothSerial.h index d59fbf1f714..8cb6edb876c 100644 --- a/libraries/BluetoothSerial/src/BluetoothSerial.h +++ b/libraries/BluetoothSerial/src/BluetoothSerial.h @@ -16,8 +16,9 @@ #define _BLUETOOTH_SERIAL_H_ #include "sdkconfig.h" +#include "soc/soc_caps.h" -#if defined(CONFIG_BT_ENABLED) && defined(CONFIG_BLUEDROID_ENABLED) +#if SOC_BT_SUPPORTED && defined(CONFIG_BT_ENABLED) && defined(CONFIG_BLUEDROID_ENABLED) #include "Arduino.h" #include "Stream.h" diff --git a/libraries/SimpleBLE/examples/SimpleBleDevice/ci.json b/libraries/SimpleBLE/examples/SimpleBleDevice/ci.json index d33a23f332e..3b6a150b31a 100644 --- a/libraries/SimpleBLE/examples/SimpleBleDevice/ci.json +++ b/libraries/SimpleBLE/examples/SimpleBleDevice/ci.json @@ -1,5 +1,6 @@ { "requires": [ + "CONFIG_SOC_BLE_SUPPORTED=y", "CONFIG_BT_ENABLED=y", "CONFIG_BLUEDROID_ENABLED=y" ] diff --git a/libraries/SimpleBLE/src/SimpleBLE.cpp b/libraries/SimpleBLE/src/SimpleBLE.cpp index 3c4ed915c05..3f1f2bbd1c1 100644 --- a/libraries/SimpleBLE/src/SimpleBLE.cpp +++ b/libraries/SimpleBLE/src/SimpleBLE.cpp @@ -13,8 +13,9 @@ // limitations under the License. #include "sdkconfig.h" +#include "soc/soc_caps.h" -#if defined(CONFIG_BT_ENABLED) && defined(CONFIG_BLUEDROID_ENABLED) +#if SOC_BT_SUPPORTED && defined(CONFIG_BT_ENABLED) && defined(CONFIG_BLUEDROID_ENABLED) #include "SimpleBLE.h" #include "esp32-hal-log.h" diff --git a/libraries/SimpleBLE/src/SimpleBLE.h b/libraries/SimpleBLE/src/SimpleBLE.h index 23c1cdb593f..df1ee751f10 100644 --- a/libraries/SimpleBLE/src/SimpleBLE.h +++ b/libraries/SimpleBLE/src/SimpleBLE.h @@ -16,8 +16,9 @@ #define _SIMPLE_BLE_H_ #include "sdkconfig.h" +#include "soc/soc_caps.h" -#if defined(CONFIG_BT_ENABLED) && defined(CONFIG_BLUEDROID_ENABLED) +#if SOC_BT_SUPPORTED && defined(CONFIG_BT_ENABLED) && defined(CONFIG_BLUEDROID_ENABLED) #include #include diff --git a/package/package_esp32_index.template.json b/package/package_esp32_index.template.json index a0d78ebc47e..63027704d8a 100644 --- a/package/package_esp32_index.template.json +++ b/package/package_esp32_index.template.json @@ -51,7 +51,7 @@ { "packager": "esp32", "name": "esp32-arduino-libs", - "version": "idf-master-d930a386-v1" + "version": "idf-master-38628f98-v1" }, { "packager": "esp32", @@ -104,63 +104,63 @@ "tools": [ { "name": "esp32-arduino-libs", - "version": "idf-master-d930a386-v1", + "version": "idf-master-38628f98-v1", "systems": [ { "host": "i686-mingw32", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-d930a386-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-d930a386-v1.zip", - "checksum": "SHA-256:0310daa4f08f807f2bf3babd2587c2694df64c70e367863eadf5020636b717ae", - "size": "422376381" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-38628f98-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-38628f98-v1.zip", + "checksum": "SHA-256:efc30a38cccff38c36a86fd3db78aeb13594da60ccf49bc7971b7a9f849abcdf", + "size": "398323971" }, { "host": "x86_64-mingw32", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-d930a386-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-d930a386-v1.zip", - "checksum": "SHA-256:0310daa4f08f807f2bf3babd2587c2694df64c70e367863eadf5020636b717ae", - "size": "422376381" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-38628f98-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-38628f98-v1.zip", + "checksum": "SHA-256:efc30a38cccff38c36a86fd3db78aeb13594da60ccf49bc7971b7a9f849abcdf", + "size": "398323971" }, { "host": "arm64-apple-darwin", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-d930a386-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-d930a386-v1.zip", - "checksum": "SHA-256:0310daa4f08f807f2bf3babd2587c2694df64c70e367863eadf5020636b717ae", - "size": "422376381" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-38628f98-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-38628f98-v1.zip", + "checksum": "SHA-256:efc30a38cccff38c36a86fd3db78aeb13594da60ccf49bc7971b7a9f849abcdf", + "size": "398323971" }, { "host": "x86_64-apple-darwin", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-d930a386-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-d930a386-v1.zip", - "checksum": "SHA-256:0310daa4f08f807f2bf3babd2587c2694df64c70e367863eadf5020636b717ae", - "size": "422376381" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-38628f98-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-38628f98-v1.zip", + "checksum": "SHA-256:efc30a38cccff38c36a86fd3db78aeb13594da60ccf49bc7971b7a9f849abcdf", + "size": "398323971" }, { "host": "x86_64-pc-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-d930a386-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-d930a386-v1.zip", - "checksum": "SHA-256:0310daa4f08f807f2bf3babd2587c2694df64c70e367863eadf5020636b717ae", - "size": "422376381" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-38628f98-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-38628f98-v1.zip", + "checksum": "SHA-256:efc30a38cccff38c36a86fd3db78aeb13594da60ccf49bc7971b7a9f849abcdf", + "size": "398323971" }, { "host": "i686-pc-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-d930a386-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-d930a386-v1.zip", - "checksum": "SHA-256:0310daa4f08f807f2bf3babd2587c2694df64c70e367863eadf5020636b717ae", - "size": "422376381" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-38628f98-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-38628f98-v1.zip", + "checksum": "SHA-256:efc30a38cccff38c36a86fd3db78aeb13594da60ccf49bc7971b7a9f849abcdf", + "size": "398323971" }, { "host": "aarch64-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-d930a386-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-d930a386-v1.zip", - "checksum": "SHA-256:0310daa4f08f807f2bf3babd2587c2694df64c70e367863eadf5020636b717ae", - "size": "422376381" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-38628f98-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-38628f98-v1.zip", + "checksum": "SHA-256:efc30a38cccff38c36a86fd3db78aeb13594da60ccf49bc7971b7a9f849abcdf", + "size": "398323971" }, { "host": "arm-linux-gnueabihf", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-d930a386-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-d930a386-v1.zip", - "checksum": "SHA-256:0310daa4f08f807f2bf3babd2587c2694df64c70e367863eadf5020636b717ae", - "size": "422376381" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-38628f98-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-38628f98-v1.zip", + "checksum": "SHA-256:efc30a38cccff38c36a86fd3db78aeb13594da60ccf49bc7971b7a9f849abcdf", + "size": "398323971" } ] }, From 602f1f6e7fbc56e36068c9521272d64d40e4df30 Mon Sep 17 00:00:00 2001 From: Me No Dev Date: Wed, 7 May 2025 16:24:33 +0300 Subject: [PATCH 021/173] IDF master (#11342) * fix(ci): ESP32-P4 hosted compile fail (#11341) * fix(ci): Update changes for P4 and C5 builds with latest IDF * IDF master aaebc374 * ci(pre-commit): Apply automatic fixes --------- Co-authored-by: Jason2866 <24528715+Jason2866@users.noreply.github.com> Co-authored-by: pre-commit-ci-lite[bot] <117423508+pre-commit-ci-lite[bot]@users.noreply.github.com> --- CMakeLists.txt | 2 +- cores/esp32/chip-debug-report.cpp | 7 +- idf_component.yml | 2 +- libraries/WiFi/src/WiFiGeneric.cpp | 12 +- package/package_esp32_index.template.json | 128 +++++++++++----------- 5 files changed, 81 insertions(+), 70 deletions(-) diff --git a/CMakeLists.txt b/CMakeLists.txt index 5ba9c999d81..9e981130715 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -362,7 +362,7 @@ set(priv_requires fatfs nvs_flash app_update spiffs bootloader_support bt esp_hi if(NOT CONFIG_ARDUINO_SELECTIVE_COMPILATION OR CONFIG_ARDUINO_SELECTIVE_OpenThread) #if(CONFIG_SOC_IEEE802154_SUPPORTED) # Does not work! #if(CONFIG_OPENTHREAD_ENABLED) # Does not work! - if(IDF_TARGET STREQUAL "esp32c6" OR IDF_TARGET STREQUAL "esp32h2") # Sadly only this works + if(IDF_TARGET STREQUAL "esp32c6" OR IDF_TARGET STREQUAL "esp32h2" OR IDF_TARGET STREQUAL "esp32c5") # Sadly only this works list(APPEND requires openthread) endif() endif() diff --git a/cores/esp32/chip-debug-report.cpp b/cores/esp32/chip-debug-report.cpp index 281c7bdb62d..8592031ee3f 100644 --- a/cores/esp32/chip-debug-report.cpp +++ b/cores/esp32/chip-debug-report.cpp @@ -68,7 +68,8 @@ static void printPkgVersion(void) { uint32_t pkg_ver = REG_GET_FIELD(EFUSE_RD_MAC_SYS_2_REG, EFUSE_PKG_VERSION); chip_report_printf("%lu", pkg_ver); #elif CONFIG_IDF_TARGET_ESP32C5 - uint32_t pkg_ver = REG_GET_FIELD(EFUSE_RD_MAC_SYS2_REG, EFUSE_PKG_VERSION); + // ToDo: Update this line when EFUSE_PKG_VERSION is available again for ESP32-C5 + uint32_t pkg_ver = 0; //REG_GET_FIELD(EFUSE_RD_MAC_SYS2_REG, EFUSE_PKG_VERSION); chip_report_printf("%lu", pkg_ver); #else chip_report_printf("Unknown"); @@ -92,11 +93,11 @@ static void printChipInfo(void) { case CHIP_ESP32H2: chip_report_printf("ESP32-H2\n"); break; case CHIP_ESP32P4: chip_report_printf("ESP32-P4\n"); break; #if ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 5, 0) - case CHIP_ESP32C5: chip_report_printf("ESP32-C5\n"); break; + case CHIP_ESP32C5: chip_report_printf("ESP32-C5\n"); break; case CHIP_ESP32C61: chip_report_printf("ESP32-C61\n"); break; case CHIP_ESP32H21: chip_report_printf("ESP32-H21\n"); break; #endif - default: chip_report_printf("Unknown %d\n", info.model); break; + default: chip_report_printf("Unknown %d\n", info.model); break; } printPkgVersion(); chip_report_printf(" Revision : %.2f\n", (float)(info.revision) / 100.0); diff --git a/idf_component.yml b/idf_component.yml index f0e3f84b28d..642bc54143d 100644 --- a/idf_component.yml +++ b/idf_component.yml @@ -105,7 +105,7 @@ dependencies: rules: - if: "target in [esp32s3]" espressif/esp_hosted: - version: "^0.0.25" + version: "^2.0.0" rules: - if: "target == esp32p4" espressif/esp_wifi_remote: diff --git a/libraries/WiFi/src/WiFiGeneric.cpp b/libraries/WiFi/src/WiFiGeneric.cpp index e669ba81c12..6ae451d1cb7 100644 --- a/libraries/WiFi/src/WiFiGeneric.cpp +++ b/libraries/WiFi/src/WiFiGeneric.cpp @@ -252,13 +252,23 @@ static bool wifiHostedInit() { if (!hosted_initialized) { hosted_initialized = true; struct esp_hosted_sdio_config conf = INIT_DEFAULT_HOST_SDIO_CONFIG(); +#ifdef BOARD_HAS_SDIO_ESP_HOSTED + conf.pin_clk.pin = BOARD_SDIO_ESP_HOSTED_CLK; + conf.pin_cmd.pin = BOARD_SDIO_ESP_HOSTED_CMD; + conf.pin_d0.pin = BOARD_SDIO_ESP_HOSTED_D0; + conf.pin_d1.pin = BOARD_SDIO_ESP_HOSTED_D1; + conf.pin_d2.pin = BOARD_SDIO_ESP_HOSTED_D2; + conf.pin_d3.pin = BOARD_SDIO_ESP_HOSTED_D3; + conf.pin_reset.pin = BOARD_SDIO_ESP_HOSTED_RESET; +#else conf.pin_clk.pin = CONFIG_ESP_SDIO_PIN_CLK; conf.pin_cmd.pin = CONFIG_ESP_SDIO_PIN_CMD; conf.pin_d0.pin = CONFIG_ESP_SDIO_PIN_D0; conf.pin_d1.pin = CONFIG_ESP_SDIO_PIN_D1; conf.pin_d2.pin = CONFIG_ESP_SDIO_PIN_D2; conf.pin_d3.pin = CONFIG_ESP_SDIO_PIN_D3; - //conf.pin_rst.pin = CONFIG_ESP_SDIO_GPIO_RESET_SLAVE; + conf.pin_reset.pin = CONFIG_ESP_SDIO_GPIO_RESET_SLAVE; +#endif // esp_hosted_sdio_set_config() will fail on second attempt but here temporarily to not cause exception on reinit if (esp_hosted_sdio_set_config(&conf) != ESP_OK || esp_hosted_init() != ESP_OK) { log_e("esp_hosted_init failed!"); diff --git a/package/package_esp32_index.template.json b/package/package_esp32_index.template.json index 63027704d8a..aa21a192509 100644 --- a/package/package_esp32_index.template.json +++ b/package/package_esp32_index.template.json @@ -51,7 +51,7 @@ { "packager": "esp32", "name": "esp32-arduino-libs", - "version": "idf-master-38628f98-v1" + "version": "idf-master-aaebc374-v1" }, { "packager": "esp32", @@ -76,7 +76,7 @@ { "packager": "esp32", "name": "openocd-esp32", - "version": "v0.12.0-esp32-20250226" + "version": "v0.12.0-esp32-20250422" }, { "packager": "esp32", @@ -104,63 +104,63 @@ "tools": [ { "name": "esp32-arduino-libs", - "version": "idf-master-38628f98-v1", + "version": "idf-master-aaebc374-v1", "systems": [ { "host": "i686-mingw32", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-38628f98-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-38628f98-v1.zip", - "checksum": "SHA-256:efc30a38cccff38c36a86fd3db78aeb13594da60ccf49bc7971b7a9f849abcdf", - "size": "398323971" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-aaebc374-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-aaebc374-v1.zip", + "checksum": "SHA-256:5baa0bbeae58973fd6bd4004332eb554b723a1e67f1af389840fac8081c2e21e", + "size": "404569749" }, { "host": "x86_64-mingw32", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-38628f98-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-38628f98-v1.zip", - "checksum": "SHA-256:efc30a38cccff38c36a86fd3db78aeb13594da60ccf49bc7971b7a9f849abcdf", - "size": "398323971" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-aaebc374-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-aaebc374-v1.zip", + "checksum": "SHA-256:5baa0bbeae58973fd6bd4004332eb554b723a1e67f1af389840fac8081c2e21e", + "size": "404569749" }, { "host": "arm64-apple-darwin", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-38628f98-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-38628f98-v1.zip", - "checksum": "SHA-256:efc30a38cccff38c36a86fd3db78aeb13594da60ccf49bc7971b7a9f849abcdf", - "size": "398323971" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-aaebc374-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-aaebc374-v1.zip", + "checksum": "SHA-256:5baa0bbeae58973fd6bd4004332eb554b723a1e67f1af389840fac8081c2e21e", + "size": "404569749" }, { "host": "x86_64-apple-darwin", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-38628f98-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-38628f98-v1.zip", - "checksum": "SHA-256:efc30a38cccff38c36a86fd3db78aeb13594da60ccf49bc7971b7a9f849abcdf", - "size": "398323971" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-aaebc374-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-aaebc374-v1.zip", + "checksum": "SHA-256:5baa0bbeae58973fd6bd4004332eb554b723a1e67f1af389840fac8081c2e21e", + "size": "404569749" }, { "host": "x86_64-pc-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-38628f98-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-38628f98-v1.zip", - "checksum": "SHA-256:efc30a38cccff38c36a86fd3db78aeb13594da60ccf49bc7971b7a9f849abcdf", - "size": "398323971" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-aaebc374-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-aaebc374-v1.zip", + "checksum": "SHA-256:5baa0bbeae58973fd6bd4004332eb554b723a1e67f1af389840fac8081c2e21e", + "size": "404569749" }, { "host": "i686-pc-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-38628f98-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-38628f98-v1.zip", - "checksum": "SHA-256:efc30a38cccff38c36a86fd3db78aeb13594da60ccf49bc7971b7a9f849abcdf", - "size": "398323971" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-aaebc374-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-aaebc374-v1.zip", + "checksum": "SHA-256:5baa0bbeae58973fd6bd4004332eb554b723a1e67f1af389840fac8081c2e21e", + "size": "404569749" }, { "host": "aarch64-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-38628f98-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-38628f98-v1.zip", - "checksum": "SHA-256:efc30a38cccff38c36a86fd3db78aeb13594da60ccf49bc7971b7a9f849abcdf", - "size": "398323971" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-aaebc374-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-aaebc374-v1.zip", + "checksum": "SHA-256:5baa0bbeae58973fd6bd4004332eb554b723a1e67f1af389840fac8081c2e21e", + "size": "404569749" }, { "host": "arm-linux-gnueabihf", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-38628f98-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-38628f98-v1.zip", - "checksum": "SHA-256:efc30a38cccff38c36a86fd3db78aeb13594da60ccf49bc7971b7a9f849abcdf", - "size": "398323971" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-aaebc374-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-aaebc374-v1.zip", + "checksum": "SHA-256:5baa0bbeae58973fd6bd4004332eb554b723a1e67f1af389840fac8081c2e21e", + "size": "404569749" } ] }, @@ -414,56 +414,56 @@ }, { "name": "openocd-esp32", - "version": "v0.12.0-esp32-20250226", + "version": "v0.12.0-esp32-20250422", "systems": [ { "host": "x86_64-pc-linux-gnu", - "url": "https://github.com/espressif/openocd-esp32/releases/download/v0.12.0-esp32-20250226/openocd-esp32-linux-amd64-0.12.0-esp32-20250226.tar.gz", - "archiveFileName": "openocd-esp32-linux-amd64-0.12.0-esp32-20250226.tar.gz", - "checksum": "SHA-256:914c726342ba5828e53f41aa454f01f317c42d8e6772d3d874593a6960fc4729", - "size": "2414924" + "url": "https://github.com/espressif/openocd-esp32/releases/download/v0.12.0-esp32-20250422/openocd-esp32-linux-amd64-0.12.0-esp32-20250422.tar.gz", + "archiveFileName": "openocd-esp32-linux-amd64-0.12.0-esp32-20250422.tar.gz", + "checksum": "SHA-256:eb1fa9b21c65b45a2200af6dcc2914e32335d37b6dbbd181778dcc0dc025e70a", + "size": "2445546" }, { "host": "aarch64-linux-gnu", - "url": "https://github.com/espressif/openocd-esp32/releases/download/v0.12.0-esp32-20250226/openocd-esp32-linux-arm64-0.12.0-esp32-20250226.tar.gz", - "archiveFileName": "openocd-esp32-linux-arm64-0.12.0-esp32-20250226.tar.gz", - "checksum": "SHA-256:c44ee99a9209c0234dbbcec86339fd685f5c61a763b29c33eba590bf62db2296", - "size": "2293923" + "url": "https://github.com/espressif/openocd-esp32/releases/download/v0.12.0-esp32-20250422/openocd-esp32-linux-arm64-0.12.0-esp32-20250422.tar.gz", + "archiveFileName": "openocd-esp32-linux-arm64-0.12.0-esp32-20250422.tar.gz", + "checksum": "SHA-256:f70334a9b12a75b4d943e09fa5db30973037c39dbb54d6fa9f1a7118228b3d1c", + "size": "2330926" }, { "host": "arm-linux-gnueabihf", - "url": "https://github.com/espressif/openocd-esp32/releases/download/v0.12.0-esp32-20250226/openocd-esp32-linux-armel-0.12.0-esp32-20250226.tar.gz", - "archiveFileName": "openocd-esp32-linux-armel-0.12.0-esp32-20250226.tar.gz", - "checksum": "SHA-256:21ab6af3cf05f9290f4d59f1f381d5094dd2755fc528d3d2feb9334348fc0d8d", - "size": "2436071" + "url": "https://github.com/espressif/openocd-esp32/releases/download/v0.12.0-esp32-20250422/openocd-esp32-linux-armel-0.12.0-esp32-20250422.tar.gz", + "archiveFileName": "openocd-esp32-linux-armel-0.12.0-esp32-20250422.tar.gz", + "checksum": "SHA-256:4ac34d6fd1af86aeda87c8318732f8d691c300c285c7fd2f5037c432c63fbbb3", + "size": "2470732" }, { "host": "x86_64-apple-darwin", - "url": "https://github.com/espressif/openocd-esp32/releases/download/v0.12.0-esp32-20250226/openocd-esp32-macos-0.12.0-esp32-20250226.tar.gz", - "archiveFileName": "openocd-esp32-macos-0.12.0-esp32-20250226.tar.gz", - "checksum": "SHA-256:0b5751699e93b6d101381611c96216ddff8c7dfd16425c610993fa27993f9a0a", - "size": "2525387" + "url": "https://github.com/espressif/openocd-esp32/releases/download/v0.12.0-esp32-20250422/openocd-esp32-macos-0.12.0-esp32-20250422.tar.gz", + "archiveFileName": "openocd-esp32-macos-0.12.0-esp32-20250422.tar.gz", + "checksum": "SHA-256:9186a7a06304c6d9201cbce4ee3c7099b393bf8d329cda17a68874f92308f6ce", + "size": "2548730" }, { "host": "arm64-apple-darwin", - "url": "https://github.com/espressif/openocd-esp32/releases/download/v0.12.0-esp32-20250226/openocd-esp32-macos-arm64-0.12.0-esp32-20250226.tar.gz", - "archiveFileName": "openocd-esp32-macos-arm64-0.12.0-esp32-20250226.tar.gz", - "checksum": "SHA-256:8bffbbb594b27a4971a3922792135f8c836fff26991f7f450094386920263531", - "size": "2568843" + "url": "https://github.com/espressif/openocd-esp32/releases/download/v0.12.0-esp32-20250422/openocd-esp32-macos-arm64-0.12.0-esp32-20250422.tar.gz", + "archiveFileName": "openocd-esp32-macos-arm64-0.12.0-esp32-20250422.tar.gz", + "checksum": "SHA-256:2cc39318d52f393233ff1f777871aebe5b97b3fbad29556a238489263401b774", + "size": "2593819" }, { "host": "i686-mingw32", - "url": "https://github.com/espressif/openocd-esp32/releases/download/v0.12.0-esp32-20250226/openocd-esp32-win32-0.12.0-esp32-20250226.zip", - "archiveFileName": "openocd-esp32-win32-0.12.0-esp32-20250226.zip", - "checksum": "SHA-256:aaf3c955bb4eb47805a1ba108dfd07a8a56ce720cb40194a354362b5f0961230", - "size": "2960226" + "url": "https://github.com/espressif/openocd-esp32/releases/download/v0.12.0-esp32-20250422/openocd-esp32-win32-0.12.0-esp32-20250422.zip", + "archiveFileName": "openocd-esp32-win32-0.12.0-esp32-20250422.zip", + "checksum": "SHA-256:ecb4f8533fa9098d10000f5f7e8b8eaa8591015b824b481078ddb2b37e7aa6f2", + "size": "2988859" }, { "host": "x86_64-mingw32", - "url": "https://github.com/espressif/openocd-esp32/releases/download/v0.12.0-esp32-20250226/openocd-esp32-win64-0.12.0-esp32-20250226.zip", - "archiveFileName": "openocd-esp32-win64-0.12.0-esp32-20250226.zip", - "checksum": "SHA-256:79baf35325117a53093b62f6b9bee677dd12275d7066e3f8a274d2a80e986b6e", - "size": "2960225" + "url": "https://github.com/espressif/openocd-esp32/releases/download/v0.12.0-esp32-20250422/openocd-esp32-win64-0.12.0-esp32-20250422.zip", + "archiveFileName": "openocd-esp32-win64-0.12.0-esp32-20250422.zip", + "checksum": "SHA-256:e9eae8e1a8d0e030cd81dcb08394a9137cb7338a6211dfabcdbdfb37b58c5a23", + "size": "2988858" } ] }, From bc08c49579d083503d8db77254d62dcb36f0057f Mon Sep 17 00:00:00 2001 From: Jason2866 <24528715+Jason2866@users.noreply.github.com> Date: Tue, 13 May 2025 12:25:39 +0200 Subject: [PATCH 022/173] Update version for wifi remote (#11344) --- idf_component.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/idf_component.yml b/idf_component.yml index 642bc54143d..98dd5c9b9d9 100644 --- a/idf_component.yml +++ b/idf_component.yml @@ -109,7 +109,7 @@ dependencies: rules: - if: "target == esp32p4" espressif/esp_wifi_remote: - version: "^0.5.4" + version: "~0.9.2" rules: - if: "target == esp32p4" espressif/libsodium: From 69cd5a0a664e96afb7007efe061bc80fef6f72cd Mon Sep 17 00:00:00 2001 From: Me No Dev Date: Thu, 15 May 2025 11:10:00 +0200 Subject: [PATCH 023/173] IDF master 7cf5dacd (#11358) --- package/package_esp32_index.template.json | 68 +++++++++++------------ 1 file changed, 34 insertions(+), 34 deletions(-) diff --git a/package/package_esp32_index.template.json b/package/package_esp32_index.template.json index aa21a192509..e3b42431cbf 100644 --- a/package/package_esp32_index.template.json +++ b/package/package_esp32_index.template.json @@ -51,7 +51,7 @@ { "packager": "esp32", "name": "esp32-arduino-libs", - "version": "idf-master-aaebc374-v1" + "version": "idf-master-7cf5dacd-v2" }, { "packager": "esp32", @@ -104,63 +104,63 @@ "tools": [ { "name": "esp32-arduino-libs", - "version": "idf-master-aaebc374-v1", + "version": "idf-master-7cf5dacd-v2", "systems": [ { "host": "i686-mingw32", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-aaebc374-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-aaebc374-v1.zip", - "checksum": "SHA-256:5baa0bbeae58973fd6bd4004332eb554b723a1e67f1af389840fac8081c2e21e", - "size": "404569749" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-7cf5dacd-v2.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-7cf5dacd-v2.zip", + "checksum": "SHA-256:91f74810c5bee0fa4463f7df73f1f6b5e4a08a6c2094014b9a4c0fac61eb43de", + "size": "405095217" }, { "host": "x86_64-mingw32", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-aaebc374-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-aaebc374-v1.zip", - "checksum": "SHA-256:5baa0bbeae58973fd6bd4004332eb554b723a1e67f1af389840fac8081c2e21e", - "size": "404569749" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-7cf5dacd-v2.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-7cf5dacd-v2.zip", + "checksum": "SHA-256:91f74810c5bee0fa4463f7df73f1f6b5e4a08a6c2094014b9a4c0fac61eb43de", + "size": "405095217" }, { "host": "arm64-apple-darwin", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-aaebc374-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-aaebc374-v1.zip", - "checksum": "SHA-256:5baa0bbeae58973fd6bd4004332eb554b723a1e67f1af389840fac8081c2e21e", - "size": "404569749" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-7cf5dacd-v2.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-7cf5dacd-v2.zip", + "checksum": "SHA-256:91f74810c5bee0fa4463f7df73f1f6b5e4a08a6c2094014b9a4c0fac61eb43de", + "size": "405095217" }, { "host": "x86_64-apple-darwin", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-aaebc374-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-aaebc374-v1.zip", - "checksum": "SHA-256:5baa0bbeae58973fd6bd4004332eb554b723a1e67f1af389840fac8081c2e21e", - "size": "404569749" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-7cf5dacd-v2.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-7cf5dacd-v2.zip", + "checksum": "SHA-256:91f74810c5bee0fa4463f7df73f1f6b5e4a08a6c2094014b9a4c0fac61eb43de", + "size": "405095217" }, { "host": "x86_64-pc-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-aaebc374-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-aaebc374-v1.zip", - "checksum": "SHA-256:5baa0bbeae58973fd6bd4004332eb554b723a1e67f1af389840fac8081c2e21e", - "size": "404569749" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-7cf5dacd-v2.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-7cf5dacd-v2.zip", + "checksum": "SHA-256:91f74810c5bee0fa4463f7df73f1f6b5e4a08a6c2094014b9a4c0fac61eb43de", + "size": "405095217" }, { "host": "i686-pc-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-aaebc374-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-aaebc374-v1.zip", - "checksum": "SHA-256:5baa0bbeae58973fd6bd4004332eb554b723a1e67f1af389840fac8081c2e21e", - "size": "404569749" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-7cf5dacd-v2.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-7cf5dacd-v2.zip", + "checksum": "SHA-256:91f74810c5bee0fa4463f7df73f1f6b5e4a08a6c2094014b9a4c0fac61eb43de", + "size": "405095217" }, { "host": "aarch64-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-aaebc374-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-aaebc374-v1.zip", - "checksum": "SHA-256:5baa0bbeae58973fd6bd4004332eb554b723a1e67f1af389840fac8081c2e21e", - "size": "404569749" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-7cf5dacd-v2.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-7cf5dacd-v2.zip", + "checksum": "SHA-256:91f74810c5bee0fa4463f7df73f1f6b5e4a08a6c2094014b9a4c0fac61eb43de", + "size": "405095217" }, { "host": "arm-linux-gnueabihf", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-aaebc374-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-aaebc374-v1.zip", - "checksum": "SHA-256:5baa0bbeae58973fd6bd4004332eb554b723a1e67f1af389840fac8081c2e21e", - "size": "404569749" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-7cf5dacd-v2.zip", + "archiveFileName": "esp32-arduino-libs-idf-master-7cf5dacd-v2.zip", + "checksum": "SHA-256:91f74810c5bee0fa4463f7df73f1f6b5e4a08a6c2094014b9a4c0fac61eb43de", + "size": "405095217" } ] }, From 5540afa1ea9ab070d85e1d4fa3338e6c6ce12405 Mon Sep 17 00:00:00 2001 From: me-no-dev Date: Thu, 15 May 2025 11:13:32 +0200 Subject: [PATCH 024/173] feat(ci): Run push CI against IDF v5.5 --- .github/workflows/push.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/push.yml b/.github/workflows/push.yml index 026e0b8fc0d..68148fbcff7 100644 --- a/.github/workflows/push.yml +++ b/.github/workflows/push.yml @@ -248,7 +248,7 @@ jobs: # See https://hub.docker.com/r/espressif/idf/tags and # https://docs.espressif.com/projects/esp-idf/en/latest/esp32/api-guides/tools/idf-docker-image.html # for details. - idf_ver: ["release-v5.4"] + idf_ver: ["release-v5.5"] idf_target: [ "esp32", From 3f63a4929f6cb7165e78d47c4478f62a76e00e9e Mon Sep 17 00:00:00 2001 From: me-no-dev Date: Mon, 19 May 2025 13:54:12 +0200 Subject: [PATCH 025/173] fix(build): Add dependency on esp_http_client --- CMakeLists.txt | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/CMakeLists.txt b/CMakeLists.txt index 9e981130715..3288f16f39f 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -356,7 +356,7 @@ endforeach() set(includedirs variants/${CONFIG_ARDUINO_VARIANT}/ cores/esp32/ ${ARDUINO_LIBRARIES_INCLUDEDIRS}) set(srcs ${CORE_SRCS} ${ARDUINO_LIBRARIES_SRCS}) set(priv_includes cores/esp32/libb64) -set(requires spi_flash esp_partition mbedtls wpa_supplicant esp_adc esp_eth http_parser esp_ringbuf esp_driver_gptimer esp_driver_usb_serial_jtag driver) +set(requires spi_flash esp_partition mbedtls wpa_supplicant esp_adc esp_eth http_parser esp_ringbuf esp_driver_gptimer esp_driver_usb_serial_jtag driver esp_http_client) set(priv_requires fatfs nvs_flash app_update spiffs bootloader_support bt esp_hid usb esp_psram ${ARDUINO_LIBRARIES_REQUIRES}) if(NOT CONFIG_ARDUINO_SELECTIVE_COMPILATION OR CONFIG_ARDUINO_SELECTIVE_OpenThread) From 6ce7e254c43604c4927ef1bcca0468e1d33db243 Mon Sep 17 00:00:00 2001 From: Me No Dev Date: Fri, 30 May 2025 17:18:32 +0300 Subject: [PATCH 026/173] IDF release/v5.5 (#11369) * IDF release/v5.5 719b1b2b * fix(build): Add dependency on esp_http_client * fix(build): Add dependency on esp_https_ota * IDF release/v5.5 28ac0243 --- CMakeLists.txt | 2 +- package/package_esp32_index.template.json | 68 +++++++++++------------ 2 files changed, 35 insertions(+), 35 deletions(-) diff --git a/CMakeLists.txt b/CMakeLists.txt index 7cee7962a65..3cbb59dd3d9 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -359,7 +359,7 @@ endforeach() set(includedirs variants/${CONFIG_ARDUINO_VARIANT}/ cores/esp32/ ${ARDUINO_LIBRARIES_INCLUDEDIRS}) set(srcs ${CORE_SRCS} ${ARDUINO_LIBRARIES_SRCS}) set(priv_includes cores/esp32/libb64) -set(requires spi_flash esp_partition mbedtls wpa_supplicant esp_adc esp_eth http_parser esp_ringbuf esp_driver_gptimer esp_driver_usb_serial_jtag driver esp_http_client) +set(requires spi_flash esp_partition mbedtls wpa_supplicant esp_adc esp_eth http_parser esp_ringbuf esp_driver_gptimer esp_driver_usb_serial_jtag driver esp_http_client esp_https_ota) set(priv_requires fatfs nvs_flash app_update spiffs bootloader_support bt esp_hid usb esp_psram ${ARDUINO_LIBRARIES_REQUIRES}) if(NOT CONFIG_ARDUINO_SELECTIVE_COMPILATION OR CONFIG_ARDUINO_SELECTIVE_OpenThread) diff --git a/package/package_esp32_index.template.json b/package/package_esp32_index.template.json index e3b42431cbf..eecc7c10788 100644 --- a/package/package_esp32_index.template.json +++ b/package/package_esp32_index.template.json @@ -51,7 +51,7 @@ { "packager": "esp32", "name": "esp32-arduino-libs", - "version": "idf-master-7cf5dacd-v2" + "version": "idf-release_v5.5-28ac0243-v1" }, { "packager": "esp32", @@ -104,63 +104,63 @@ "tools": [ { "name": "esp32-arduino-libs", - "version": "idf-master-7cf5dacd-v2", + "version": "idf-release_v5.5-28ac0243-v1", "systems": [ { "host": "i686-mingw32", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-7cf5dacd-v2.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-7cf5dacd-v2.zip", - "checksum": "SHA-256:91f74810c5bee0fa4463f7df73f1f6b5e4a08a6c2094014b9a4c0fac61eb43de", - "size": "405095217" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-28ac0243-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-28ac0243-v1.zip", + "checksum": "SHA-256:280401ea803d8a782c11ef4f96cfbf80eb12a0f51bd12eac9cb96d6c26489f6e", + "size": "405149394" }, { "host": "x86_64-mingw32", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-7cf5dacd-v2.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-7cf5dacd-v2.zip", - "checksum": "SHA-256:91f74810c5bee0fa4463f7df73f1f6b5e4a08a6c2094014b9a4c0fac61eb43de", - "size": "405095217" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-28ac0243-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-28ac0243-v1.zip", + "checksum": "SHA-256:280401ea803d8a782c11ef4f96cfbf80eb12a0f51bd12eac9cb96d6c26489f6e", + "size": "405149394" }, { "host": "arm64-apple-darwin", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-7cf5dacd-v2.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-7cf5dacd-v2.zip", - "checksum": "SHA-256:91f74810c5bee0fa4463f7df73f1f6b5e4a08a6c2094014b9a4c0fac61eb43de", - "size": "405095217" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-28ac0243-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-28ac0243-v1.zip", + "checksum": "SHA-256:280401ea803d8a782c11ef4f96cfbf80eb12a0f51bd12eac9cb96d6c26489f6e", + "size": "405149394" }, { "host": "x86_64-apple-darwin", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-7cf5dacd-v2.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-7cf5dacd-v2.zip", - "checksum": "SHA-256:91f74810c5bee0fa4463f7df73f1f6b5e4a08a6c2094014b9a4c0fac61eb43de", - "size": "405095217" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-28ac0243-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-28ac0243-v1.zip", + "checksum": "SHA-256:280401ea803d8a782c11ef4f96cfbf80eb12a0f51bd12eac9cb96d6c26489f6e", + "size": "405149394" }, { "host": "x86_64-pc-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-7cf5dacd-v2.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-7cf5dacd-v2.zip", - "checksum": "SHA-256:91f74810c5bee0fa4463f7df73f1f6b5e4a08a6c2094014b9a4c0fac61eb43de", - "size": "405095217" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-28ac0243-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-28ac0243-v1.zip", + "checksum": "SHA-256:280401ea803d8a782c11ef4f96cfbf80eb12a0f51bd12eac9cb96d6c26489f6e", + "size": "405149394" }, { "host": "i686-pc-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-7cf5dacd-v2.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-7cf5dacd-v2.zip", - "checksum": "SHA-256:91f74810c5bee0fa4463f7df73f1f6b5e4a08a6c2094014b9a4c0fac61eb43de", - "size": "405095217" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-28ac0243-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-28ac0243-v1.zip", + "checksum": "SHA-256:280401ea803d8a782c11ef4f96cfbf80eb12a0f51bd12eac9cb96d6c26489f6e", + "size": "405149394" }, { "host": "aarch64-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-7cf5dacd-v2.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-7cf5dacd-v2.zip", - "checksum": "SHA-256:91f74810c5bee0fa4463f7df73f1f6b5e4a08a6c2094014b9a4c0fac61eb43de", - "size": "405095217" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-28ac0243-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-28ac0243-v1.zip", + "checksum": "SHA-256:280401ea803d8a782c11ef4f96cfbf80eb12a0f51bd12eac9cb96d6c26489f6e", + "size": "405149394" }, { "host": "arm-linux-gnueabihf", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-master/esp32-arduino-libs-idf-master-7cf5dacd-v2.zip", - "archiveFileName": "esp32-arduino-libs-idf-master-7cf5dacd-v2.zip", - "checksum": "SHA-256:91f74810c5bee0fa4463f7df73f1f6b5e4a08a6c2094014b9a4c0fac61eb43de", - "size": "405095217" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-28ac0243-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-28ac0243-v1.zip", + "checksum": "SHA-256:280401ea803d8a782c11ef4f96cfbf80eb12a0f51bd12eac9cb96d6c26489f6e", + "size": "405149394" } ] }, From a6bba43d32ab65d73054a0a65b63a4716673975c Mon Sep 17 00:00:00 2001 From: me-no-dev Date: Fri, 30 May 2025 19:12:34 +0300 Subject: [PATCH 027/173] fix(c5): Enable components for ESP32-C5 --- idf_component.yml | 18 +++++++++--------- 1 file changed, 9 insertions(+), 9 deletions(-) diff --git a/idf_component.yml b/idf_component.yml index 98dd5c9b9d9..dca97dc1655 100644 --- a/idf_component.yml +++ b/idf_component.yml @@ -57,12 +57,12 @@ dependencies: version: "==1.6.3" require: public rules: - - if: "target not in [esp32c2, esp32p4, esp32c5]" + - if: "target not in [esp32c2, esp32p4]" espressif/esp-zigbee-lib: version: "==1.6.3" require: public rules: - - if: "target not in [esp32c2, esp32p4, esp32c5]" + - if: "target not in [esp32c2, esp32p4]" espressif/esp-dsp: version: "^1.3.4" rules: @@ -73,32 +73,32 @@ dependencies: espressif/esp_rainmaker: version: "1.5.2" rules: - - if: "target not in [esp32c2, esp32p4, esp32c5]" + - if: "target not in [esp32c2, esp32p4]" espressif/rmaker_common: version: "1.4.6" rules: - - if: "target not in [esp32c2, esp32p4, esp32c5]" + - if: "target not in [esp32c2, esp32p4]" espressif/esp_insights: version: "1.2.2" rules: - - if: "target not in [esp32c2, esp32p4, esp32c5]" + - if: "target not in [esp32c2, esp32p4]" # New version breaks esp_insights 1.0.1 espressif/esp_diag_data_store: version: "1.0.2" rules: - - if: "target not in [esp32c2, esp32p4, esp32c5]" + - if: "target not in [esp32c2, esp32p4]" espressif/esp_diagnostics: version: "1.2.1" rules: - - if: "target not in [esp32c2, esp32p4, esp32c5]" + - if: "target not in [esp32c2, esp32p4]" espressif/cbor: version: "0.6.0~1" rules: - - if: "target not in [esp32c2, esp32p4, esp32c5]" + - if: "target not in [esp32c2, esp32p4]" espressif/qrcode: version: "0.1.0~2" rules: - - if: "target not in [esp32c2, esp32p4, esp32c5]" + - if: "target not in [esp32c2, esp32p4]" # RainMaker End espressif/esp-sr: version: "^1.4.2" From 6f56df2a09a17213823e2edda3cdff3db32cadaf Mon Sep 17 00:00:00 2001 From: Lucas Saavedra Vaz <32426024+lucasssvaz@users.noreply.github.com> Date: Wed, 4 Jun 2025 10:19:44 -0300 Subject: [PATCH 028/173] feat(codeowners): Initial CODEOWNERS setup (#11397) * feat(codeowners): Initial CODEOWNERS setup * fix(comment): Improve comment * fix(codeowners): Add teams * fix(codeowners): Apply suggestions * fix(codeowners): Add missing libraries --- .github/CODEOWNERS | 79 ++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 79 insertions(+) create mode 100644 .github/CODEOWNERS diff --git a/.github/CODEOWNERS b/.github/CODEOWNERS new file mode 100644 index 00000000000..75a2b46d619 --- /dev/null +++ b/.github/CODEOWNERS @@ -0,0 +1,79 @@ +# CODEOWNERS for ESP32 Arduino Core + +# This file is used to specify the code owners for the ESP32 Arduino Core. +# Read more about CODEOWNERS: +# https://docs.github.com/en/repositories/managing-your-repositorys-settings-and-features/customizing-your-repository/about-code-owners +# Note that order matters. The last matching pattern will be used. + +# The default owners are the active developers of the ESP32 Arduino Core. +# Refrain from using @espressif/arduino-esp32 to avoid spamming non-developers with review requests. +* @espressif/arduino-devs + +# CI +/.github/ @lucasssvaz @me-no-dev @P-R-O-C-H-Y +/tests/ @lucasssvaz @P-R-O-C-H-Y + +# Tools +/tools/ @me-no-dev +/tools/pre-commit/ @lucasssvaz +/tools/add_lib.sh @P-R-O-C-H-Y + +# Pre-commit +/.* @lucasssvaz # Files in root directory that start with a dot. + +# Git Files +/.gitignore @espressif/arduino-devs +/.gitmodules @espressif/arduino-devs + +# Documentation +/docs/ @pedrominatel +/.github/ISSUE_TEMPLATE/ @pedrominatel +/.github/PULL_REQUEST_TEMPLATE.md @pedrominatel +/.readthedocs.yaml @pedrominatel +/*.md @pedrominatel + +# Boards +/variants/ @P-R-O-C-H-Y +/boards.txt @P-R-O-C-H-Y + +# Arduino as Component +/idf_component_examples/ @SuGlider +/idf_component.yml @SuGlider @me-no-dev +/CMakeLists.txt @SuGlider @me-no-dev +/Kconfig.projbuild @SuGlider @me-no-dev + +# Build System +/package.json @me-no-dev +/platform.txt @me-no-dev +/programmers.txt @me-no-dev +/package/ @me-no-dev + +# Libraries +/libraries/ArduinoOTA/ @me-no-dev +/libraries/AsyncUDP/ @me-no-dev +/libraries/BLE/ @lucasssvaz @SuGlider +/libraries/ESP_I2S/ @me-no-dev +/libraries/ESP_NOW/ @P-R-O-C-H-Y @lucasssvaz +/libraries/ESP_SR/ @me-no-dev +/libraries/ESPmDNS/ @me-no-dev +/libraries/Ethernet/ @me-no-dev +/libraries/Matter/ @SuGlider +/libraries/NetBIOS/ @me-no-dev +/libraries/Network/ @me-no-dev +/libraries/OpenThread/ @SuGlider +/libraries/PPP/ @me-no-dev +/libraries/SPI/ @me-no-dev +/libraries/Update/ @me-no-dev +/libraries/USB/ @SuGlider @me-no-dev +/libraries/WiFi/ @me-no-dev +/libraries/WiFiProv/ @me-no-dev +/libraries/Wire/ @me-no-dev +/libraries/Zigbee/ @P-R-O-C-H-Y + +# CI JSON +# Keep this after other libraries and tests to avoid being overridden. +**/ci.json @lucasssvaz + +# The CODEOWNERS file should be owned by the developers of the ESP32 Arduino Core. +# Leave this entry as the last one to avoid being overridden. +/.github/CODEOWNERS @espressif/arduino-devs From b5c5655cf004858ed56f3488f006ebad3690c303 Mon Sep 17 00:00:00 2001 From: Kevin Sidwar Date: Wed, 4 Jun 2025 08:39:19 -0600 Subject: [PATCH 029/173] Support HTTP 204 (#11408) HTTP 204 is a successful return code which indicates No Content. While it's appropriate to return a 304 if the server has content for a device but it hasn't change, it is more accurate for a server to return a 204 if it simply doesn't have any firmware files for a particular device. Co-authored-by: Me No Dev --- libraries/HTTPUpdate/src/HTTPUpdate.cpp | 1 + 1 file changed, 1 insertion(+) diff --git a/libraries/HTTPUpdate/src/HTTPUpdate.cpp b/libraries/HTTPUpdate/src/HTTPUpdate.cpp index 1eda0a4f7e5..b463ffe2e83 100644 --- a/libraries/HTTPUpdate/src/HTTPUpdate.cpp +++ b/libraries/HTTPUpdate/src/HTTPUpdate.cpp @@ -356,6 +356,7 @@ HTTPUpdateResult HTTPUpdate::handleUpdate(HTTPClient &http, const String ¤ log_e("Content-Length was 0 or wasn't set by Server?!\n"); } break; + case HTTP_CODE_NO_CONTENT: case HTTP_CODE_NOT_MODIFIED: ///< Not Modified (No updates) ret = HTTP_UPDATE_NO_UPDATES; From a2880a4c17038b4ff3dfa083e7af7c7bbdba914a Mon Sep 17 00:00:00 2001 From: Me No Dev Date: Wed, 4 Jun 2025 17:39:34 +0300 Subject: [PATCH 030/173] feat(ap): Add support for DHCP Captive Portal (opt 114) (#11412) * feat(ap): Add support for DHCP Captive Portal (opt 114) * feat(ap): No need to guard the function * ci(pre-commit): Apply automatic fixes --------- Co-authored-by: pre-commit-ci-lite[bot] <117423508+pre-commit-ci-lite[bot]@users.noreply.github.com> --- .../examples/CaptivePortal/CaptivePortal.ino | 5 ++- libraries/WiFi/src/AP.cpp | 39 +++++++++++++++++++ libraries/WiFi/src/WiFiAP.h | 1 + 3 files changed, 43 insertions(+), 2 deletions(-) diff --git a/libraries/DNSServer/examples/CaptivePortal/CaptivePortal.ino b/libraries/DNSServer/examples/CaptivePortal/CaptivePortal.ino index d956dc14ad3..7759aa09a1c 100644 --- a/libraries/DNSServer/examples/CaptivePortal/CaptivePortal.ino +++ b/libraries/DNSServer/examples/CaptivePortal/CaptivePortal.ino @@ -34,8 +34,9 @@ void handleNotFound() { void setup() { Serial.begin(115200); - WiFi.mode(WIFI_AP); - WiFi.softAP("ESP32-DNSServer"); + WiFi.AP.begin(); + WiFi.AP.create("ESP32-DNSServer"); + WiFi.AP.enableDhcpCaptivePortal(); // by default DNSServer is started serving any "*" domain name. It will reply // AccessPoint's IP to all DNS request (this is required for Captive Portal detection) diff --git a/libraries/WiFi/src/AP.cpp b/libraries/WiFi/src/AP.cpp index 0e7839764ea..a649c3898cb 100644 --- a/libraries/WiFi/src/AP.cpp +++ b/libraries/WiFi/src/AP.cpp @@ -305,6 +305,45 @@ bool APClass::enableNAPT(bool enable) { return true; } +bool APClass::enableDhcpCaptivePortal() { + esp_err_t err = ESP_OK; + static char captiveportal_uri[32] = { + 0, + }; + + if (!started()) { + log_e("AP must be first started to enable DHCP Captive Portal"); + return false; + } + + // Create Captive Portal URL: http://192.168.0.4 + strcpy(captiveportal_uri, "http://"); + strcat(captiveportal_uri, String(localIP()).c_str()); + + // Stop DHCPS + err = esp_netif_dhcps_stop(_esp_netif); + if (err && err != ESP_ERR_ESP_NETIF_DHCP_ALREADY_STOPPED) { + log_e("DHCPS Stop Failed! 0x%04x: %s", err, esp_err_to_name(err)); + return false; + } + + // Enable DHCP Captive Portal + err = esp_netif_dhcps_option(_esp_netif, ESP_NETIF_OP_SET, ESP_NETIF_CAPTIVEPORTAL_URI, captiveportal_uri, strlen(captiveportal_uri)); + if (err) { + log_e("Could not set enable DHCP Captive Portal! 0x%x: %s", err, esp_err_to_name(err)); + return false; + } + + // Start DHCPS + err = esp_netif_dhcps_start(_esp_netif); + if (err) { + log_e("DHCPS Start Failed! 0x%04x: %s", err, esp_err_to_name(err)); + return false; + } + + return true; +} + String APClass::SSID(void) const { if (!started()) { return String(); diff --git a/libraries/WiFi/src/WiFiAP.h b/libraries/WiFi/src/WiFiAP.h index 540ec87f44f..2b7ce469801 100644 --- a/libraries/WiFi/src/WiFiAP.h +++ b/libraries/WiFi/src/WiFiAP.h @@ -53,6 +53,7 @@ class APClass : public NetworkInterface { bool bandwidth(wifi_bandwidth_t bandwidth); bool enableNAPT(bool enable = true); + bool enableDhcpCaptivePortal(); String SSID(void) const; uint8_t stationCount(); From 460b89201b0be121c4345549551bcf102c26f165 Mon Sep 17 00:00:00 2001 From: Jason2866 <24528715+Jason2866@users.noreply.github.com> Date: Wed, 4 Jun 2025 16:39:54 +0200 Subject: [PATCH 031/173] make BT core code execution conditional from include esp_bt.h (#11413) * make code execution conditional from include esp_bt.h.h * only one if * Update esp32-hal-bt.c --- cores/esp32/esp32-hal-bt.c | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/cores/esp32/esp32-hal-bt.c b/cores/esp32/esp32-hal-bt.c index 1e5f73e324c..5d512d448a3 100644 --- a/cores/esp32/esp32-hal-bt.c +++ b/cores/esp32/esp32-hal-bt.c @@ -15,7 +15,7 @@ #include "esp32-hal-bt.h" #if SOC_BT_SUPPORTED -#ifdef CONFIG_BT_BLUEDROID_ENABLED +#if defined(CONFIG_BT_BLUEDROID_ENABLED) && __has_include("esp_bt.h") #if CONFIG_IDF_TARGET_ESP32 bool btInUse() { From 375f2c002de0d5ace08bac59a4e42134be1af22e Mon Sep 17 00:00:00 2001 From: Jason2866 <24528715+Jason2866@users.noreply.github.com> Date: Wed, 4 Jun 2025 16:40:12 +0200 Subject: [PATCH 032/173] C2: Disable network provisioning (#11423) --- idf_component.yml | 2 ++ 1 file changed, 2 insertions(+) diff --git a/idf_component.yml b/idf_component.yml index 967c4ecf0f6..7f090cb26a0 100644 --- a/idf_component.yml +++ b/idf_component.yml @@ -68,6 +68,8 @@ dependencies: # RainMaker Start (Fixed versions, because Matter supports only Insights 1.0.1) espressif/network_provisioning: version: "1.0.2" + rules: + - if: "target != esp32c2" espressif/esp_rainmaker: version: "1.5.2" rules: From cae66e65a30d5a3d7c15b85ffa4130fa74e01b73 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jan=20Proch=C3=A1zka?= <90197375+P-R-O-C-H-Y@users.noreply.github.com> Date: Wed, 4 Jun 2025 16:40:28 +0200 Subject: [PATCH 033/173] feat(zigbee): Add endpoint identification in read handlers + command structures fix (#11425) * feat(zigbee): Add endpoint identification in read handlers * fix(zigbee): initialize Zigbee command structures with zeros * fix(zigbee): Spelling correction * ci(pre-commit): Apply automatic fixes --------- Co-authored-by: pre-commit-ci-lite[bot] <117423508+pre-commit-ci-lite[bot]@users.noreply.github.com> --- .../Zigbee_Thermostat/Zigbee_Thermostat.ino | 56 +++- libraries/Zigbee/keywords.txt | 3 +- libraries/Zigbee/src/ZigbeeEP.cpp | 20 +- libraries/Zigbee/src/ZigbeeEP.h | 4 +- libraries/Zigbee/src/ZigbeeHandlers.cpp | 8 +- .../Zigbee/src/ep/ZigbeeColorDimmerSwitch.cpp | 48 +-- libraries/Zigbee/src/ep/ZigbeeSwitch.cpp | 32 +- libraries/Zigbee/src/ep/ZigbeeThermostat.cpp | 279 +++++++++++++++++- libraries/Zigbee/src/ep/ZigbeeThermostat.h | 29 +- 9 files changed, 390 insertions(+), 89 deletions(-) diff --git a/libraries/Zigbee/examples/Zigbee_Thermostat/Zigbee_Thermostat.ino b/libraries/Zigbee/examples/Zigbee_Thermostat/Zigbee_Thermostat.ino index 7cdf45ef711..6f5934f791d 100644 --- a/libraries/Zigbee/examples/Zigbee_Thermostat/Zigbee_Thermostat.ino +++ b/libraries/Zigbee/examples/Zigbee_Thermostat/Zigbee_Thermostat.ino @@ -34,7 +34,8 @@ #include "Zigbee.h" /* Zigbee thermostat configuration */ -#define THERMOSTAT_ENDPOINT_NUMBER 5 +#define THERMOSTAT_ENDPOINT_NUMBER 1 +#define USE_RECEIVE_TEMP_WITH_SOURCE 1 uint8_t button = BOOT_PIN; ZigbeeThermostat zbThermostat = ZigbeeThermostat(THERMOSTAT_ENDPOINT_NUMBER); @@ -48,13 +49,28 @@ float sensor_tolerance; struct tm timeinfo = {}; // Time structure for Time cluster /****************** Temperature sensor handling *******************/ -void recieveSensorTemp(float temperature) { +#if USE_RECEIVE_TEMP_WITH_SOURCE == 0 +void receiveSensorTemp(float temperature) { Serial.printf("Temperature sensor value: %.2f°C\n", temperature); sensor_temp = temperature; } +#else +void receiveSensorTempWithSource(float temperature, uint8_t src_endpoint, esp_zb_zcl_addr_t src_address) { + if (src_address.addr_type == ESP_ZB_ZCL_ADDR_TYPE_SHORT) { + Serial.printf("Temperature sensor value: %.2f°C from endpoint %d, address 0x%04x\n", temperature, src_endpoint, src_address.u.short_addr); + } else { + Serial.printf( + "Temperature sensor value: %.2f°C from endpoint %d, address %02x:%02x:%02x:%02x:%02x:%02x:%02x:%02x\n", temperature, src_endpoint, + src_address.u.ieee_addr[7], src_address.u.ieee_addr[6], src_address.u.ieee_addr[5], src_address.u.ieee_addr[4], src_address.u.ieee_addr[3], + src_address.u.ieee_addr[2], src_address.u.ieee_addr[1], src_address.u.ieee_addr[0] + ); + } + sensor_temp = temperature; +} +#endif -void recieveSensorConfig(float min_temp, float max_temp, float tolerance) { - Serial.printf("Temperature sensor settings: min %.2f°C, max %.2f°C, tolerance %.2f°C\n", min_temp, max_temp, tolerance); +void receiveSensorConfig(float min_temp, float max_temp, float tolerance) { + Serial.printf("Temperature sensor config: min %.2f°C, max %.2f°C, tolerance %.2f°C\n", min_temp, max_temp, tolerance); sensor_min_temp = min_temp; sensor_max_temp = max_temp; sensor_tolerance = tolerance; @@ -66,9 +82,15 @@ void setup() { // Init button switch pinMode(button, INPUT_PULLUP); - // Set callback functions for temperature and configuration receive - zbThermostat.onTempRecieve(recieveSensorTemp); - zbThermostat.onConfigRecieve(recieveSensorConfig); +// Set callback function for receiving temperature from sensor - Use only one option +#if USE_RECEIVE_TEMP_WITH_SOURCE == 0 + zbThermostat.onTempReceive(receiveSensorTemp); // If you bound only one sensor or you don't need to know the source of the temperature +#else + zbThermostat.onTempReceiveWithSource(receiveSensorTempWithSource); +#endif + + // Set callback function for receiving sensor configuration + zbThermostat.onConfigReceive(receiveSensorConfig); //Optional: set Zigbee device name and model zbThermostat.setManufacturerAndModel("Espressif", "ZigbeeThermostat"); @@ -107,19 +129,30 @@ void setup() { Serial.println(); - // Get temperature sensor configuration - zbThermostat.getSensorSettings(); + // Get temperature sensor configuration for all bound sensors by endpoint number and address + std::list boundSensors = zbThermostat.getBoundDevices(); + for (const auto &device : boundSensors) { + Serial.println("--------------------------------"); + if (device->short_addr == 0x0000 || device->short_addr == 0xFFFF) { //End devices never have 0x0000 short address or 0xFFFF group address + Serial.printf( + "Device on endpoint %d, IEEE Address: %02X:%02X:%02X:%02X:%02X:%02X:%02X:%02X\r\n", device->endpoint, device->ieee_addr[7], device->ieee_addr[6], + device->ieee_addr[5], device->ieee_addr[4], device->ieee_addr[3], device->ieee_addr[2], device->ieee_addr[1], device->ieee_addr[0] + ); + zbThermostat.getSensorSettings(device->endpoint, device->ieee_addr); + } else { + Serial.printf("Device on endpoint %d, short address: 0x%x\r\n", device->endpoint, device->short_addr); + zbThermostat.getSensorSettings(device->endpoint, device->short_addr); + } + } } void loop() { // Handle button switch in loop() if (digitalRead(button) == LOW) { // Push button pressed - // Key debounce handling while (digitalRead(button) == LOW) { delay(50); } - // Set reporting interval for temperature sensor zbThermostat.setTemperatureReporting(0, 10, 2); } @@ -130,5 +163,6 @@ void loop() { last_print = millis(); int temp_percent = (int)((sensor_temp - sensor_min_temp) / (sensor_max_temp - sensor_min_temp) * 100); Serial.printf("Loop temperature info: %.2f°C (%d %%)\n", sensor_temp, temp_percent); + zbThermostat.printBoundDevices(Serial); } } diff --git a/libraries/Zigbee/keywords.txt b/libraries/Zigbee/keywords.txt index 4d4cd7d0606..5ce1e8f6f51 100644 --- a/libraries/Zigbee/keywords.txt +++ b/libraries/Zigbee/keywords.txt @@ -107,6 +107,7 @@ setLightColor KEYWORD2 # ZigbeeThermostat onTempRecieve KEYWORD2 onConfigRecieve KEYWORD2 +onTempReceiveWithSource KEYWORD2 getTemperature KEYWORD2 getSensorSettings KEYWORD2 setTemperatureReporting KEYWORD2 @@ -191,4 +192,4 @@ ZIGBEE_DEFAULT_COORDINATOR_CONFIG LITERAL1 ZIGBEE_DEFAULT_RADIO_CONFIG LITERAL1 ZIGBEE_DEFAULT_UART_RCP_RADIO_CONFIG LITERAL1 ZIGBEE_DEFAULT_HOST_CONFIG LITERAL1 -ZB_ARRAY_LENTH LITERAL1 +ZB_ARRAY_LENGHT LITERAL1 diff --git a/libraries/Zigbee/src/ZigbeeEP.cpp b/libraries/Zigbee/src/ZigbeeEP.cpp index 1d3df126ce8..efddbdd0368 100644 --- a/libraries/Zigbee/src/ZigbeeEP.cpp +++ b/libraries/Zigbee/src/ZigbeeEP.cpp @@ -145,7 +145,7 @@ bool ZigbeeEP::setBatteryVoltage(uint8_t voltage) { bool ZigbeeEP::reportBatteryPercentage() { /* Send report attributes command */ - esp_zb_zcl_report_attr_cmd_t report_attr_cmd; + esp_zb_zcl_report_attr_cmd_t report_attr_cmd = {0}; report_attr_cmd.address_mode = ESP_ZB_APS_ADDR_MODE_DST_ADDR_ENDP_NOT_PRESENT; report_attr_cmd.attributeID = ESP_ZB_ZCL_ATTR_POWER_CONFIG_BATTERY_PERCENTAGE_REMAINING_ID; report_attr_cmd.direction = ESP_ZB_ZCL_CMD_DIRECTION_TO_CLI; @@ -166,7 +166,7 @@ bool ZigbeeEP::reportBatteryPercentage() { char *ZigbeeEP::readManufacturer(uint8_t endpoint, uint16_t short_addr, esp_zb_ieee_addr_t ieee_addr) { /* Read peer Manufacture Name & Model Identifier */ - esp_zb_zcl_read_attr_cmd_t read_req; + esp_zb_zcl_read_attr_cmd_t read_req = {0}; if (short_addr != 0) { read_req.address_mode = ESP_ZB_APS_ADDR_MODE_16_ENDP_PRESENT; @@ -183,7 +183,7 @@ char *ZigbeeEP::readManufacturer(uint8_t endpoint, uint16_t short_addr, esp_zb_i uint16_t attributes[] = { ESP_ZB_ZCL_ATTR_BASIC_MANUFACTURER_NAME_ID, }; - read_req.attr_number = ZB_ARRAY_LENTH(attributes); + read_req.attr_number = ZB_ARRAY_LENGHT(attributes); read_req.attr_field = attributes; if (_read_manufacturer != NULL) { @@ -204,7 +204,7 @@ char *ZigbeeEP::readManufacturer(uint8_t endpoint, uint16_t short_addr, esp_zb_i char *ZigbeeEP::readModel(uint8_t endpoint, uint16_t short_addr, esp_zb_ieee_addr_t ieee_addr) { /* Read peer Manufacture Name & Model Identifier */ - esp_zb_zcl_read_attr_cmd_t read_req; + esp_zb_zcl_read_attr_cmd_t read_req = {0}; if (short_addr != 0) { read_req.address_mode = ESP_ZB_APS_ADDR_MODE_16_ENDP_PRESENT; @@ -221,7 +221,7 @@ char *ZigbeeEP::readModel(uint8_t endpoint, uint16_t short_addr, esp_zb_ieee_add uint16_t attributes[] = { ESP_ZB_ZCL_ATTR_BASIC_MODEL_IDENTIFIER_ID, }; - read_req.attr_number = ZB_ARRAY_LENTH(attributes); + read_req.attr_number = ZB_ARRAY_LENGHT(attributes); read_req.attr_field = attributes; if (_read_model != NULL) { @@ -375,7 +375,7 @@ bool ZigbeeEP::setTimezone(int32_t gmt_offset) { 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; + esp_zb_zcl_read_attr_cmd_t read_req = {0}; if (short_addr >= 0) { read_req.address_mode = ESP_ZB_APS_ADDR_MODE_16_ENDP_PRESENT; @@ -386,7 +386,7 @@ tm ZigbeeEP::getTime(uint8_t endpoint, int32_t short_addr, esp_zb_ieee_addr_t ie } uint16_t attributes[] = {ESP_ZB_ZCL_ATTR_TIME_TIME_ID}; - read_req.attr_number = ZB_ARRAY_LENTH(attributes); + read_req.attr_number = ZB_ARRAY_LENGHT(attributes); read_req.attr_field = attributes; read_req.clusterID = ESP_ZB_ZCL_CLUSTER_ID_TIME; @@ -427,7 +427,7 @@ tm ZigbeeEP::getTime(uint8_t endpoint, int32_t short_addr, esp_zb_ieee_addr_t ie 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; + esp_zb_zcl_read_attr_cmd_t read_req = {0}; if (short_addr >= 0) { read_req.address_mode = ESP_ZB_APS_ADDR_MODE_16_ENDP_PRESENT; @@ -438,7 +438,7 @@ int32_t ZigbeeEP::getTimezone(uint8_t endpoint, int32_t short_addr, esp_zb_ieee_ } uint16_t attributes[] = {ESP_ZB_ZCL_ATTR_TIME_TIME_ZONE_ID}; - read_req.attr_number = ZB_ARRAY_LENTH(attributes); + read_req.attr_number = ZB_ARRAY_LENGHT(attributes); read_req.attr_field = attributes; read_req.clusterID = ESP_ZB_ZCL_CLUSTER_ID_TIME; @@ -543,7 +543,7 @@ static void findOTAServer(esp_zb_zdp_status_t zdo_status, uint16_t addr, uint8_t } void ZigbeeEP::requestOTAUpdate() { - esp_zb_zdo_match_desc_req_param_t req; + esp_zb_zdo_match_desc_req_param_t req = {0}; uint16_t cluster_list[] = {ESP_ZB_ZCL_CLUSTER_ID_OTA_UPGRADE}; /* Match the OTA server of coordinator */ diff --git a/libraries/Zigbee/src/ZigbeeEP.h b/libraries/Zigbee/src/ZigbeeEP.h index fe22f31faaf..a3217cbd066 100644 --- a/libraries/Zigbee/src/ZigbeeEP.h +++ b/libraries/Zigbee/src/ZigbeeEP.h @@ -12,7 +12,7 @@ #define ZB_CMD_TIMEOUT 10000 // 10 seconds #define OTA_UPGRADE_QUERY_INTERVAL (1 * 60) // 1 hour = 60 minutes -#define ZB_ARRAY_LENTH(arr) (sizeof(arr) / sizeof(arr[0])) +#define ZB_ARRAY_LENGHT(arr) (sizeof(arr) / sizeof(arr[0])) #define RGB_TO_XYZ(r, g, b, X, Y, Z) \ { \ @@ -131,7 +131,7 @@ class ZigbeeEP { // list of all handlers function calls, to be override by EPs implementation virtual void zbAttributeSet(const esp_zb_zcl_set_attr_value_message_t *message) {}; - virtual void zbAttributeRead(uint16_t cluster_id, const esp_zb_zcl_attribute_t *attribute) {}; + virtual void zbAttributeRead(uint16_t cluster_id, const esp_zb_zcl_attribute_t *attribute, uint8_t src_endpoint, esp_zb_zcl_addr_t src_address) {}; 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) {}; diff --git a/libraries/Zigbee/src/ZigbeeHandlers.cpp b/libraries/Zigbee/src/ZigbeeHandlers.cpp index eeeb1e8013a..5d54e459058 100644 --- a/libraries/Zigbee/src/ZigbeeHandlers.cpp +++ b/libraries/Zigbee/src/ZigbeeHandlers.cpp @@ -108,7 +108,9 @@ static esp_err_t zb_attribute_reporting_handler(const esp_zb_zcl_report_attr_mes // 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->dst_endpoint == (*it)->getEndpoint()) { - (*it)->zbAttributeRead(message->cluster, &message->attribute); //method zbAttributeRead must be implemented in specific EP class + (*it)->zbAttributeRead( + message->cluster, &message->attribute, message->src_endpoint, message->src_address + ); //method zbAttributeRead must be implemented in specific EP class } } return ESP_OK; @@ -142,7 +144,9 @@ static esp_err_t zb_cmd_read_attr_resp_handler(const esp_zb_zcl_cmd_read_attr_re } 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 + (*it)->zbAttributeRead( + message->info.cluster, &variable->attribute, message->info.src_endpoint, message->info.src_address + ); //method zbAttributeRead must be implemented in specific EP class } } variable = variable->next; diff --git a/libraries/Zigbee/src/ep/ZigbeeColorDimmerSwitch.cpp b/libraries/Zigbee/src/ep/ZigbeeColorDimmerSwitch.cpp index 935d638324f..f795ed1a3c8 100644 --- a/libraries/Zigbee/src/ep/ZigbeeColorDimmerSwitch.cpp +++ b/libraries/Zigbee/src/ep/ZigbeeColorDimmerSwitch.cpp @@ -53,7 +53,7 @@ void ZigbeeColorDimmerSwitch::findCb(esp_zb_zdp_status_t zdo_status, uint16_t ad ZigbeeColorDimmerSwitch *instance = static_cast(user_ctx); if (zdo_status == ESP_ZB_ZDP_STATUS_SUCCESS) { log_d("Found light endpoint"); - esp_zb_zdo_bind_req_param_t bind_req; + esp_zb_zdo_bind_req_param_t bind_req = {0}; zb_device_params_t *light = (zb_device_params_t *)malloc(sizeof(zb_device_params_t)); light->endpoint = endpoint; light->short_addr = addr; @@ -97,7 +97,7 @@ void ZigbeeColorDimmerSwitch::findEndpoint(esp_zb_zdo_match_desc_req_param_t *cm // Methods to control the light void ZigbeeColorDimmerSwitch::lightToggle() { if (_is_bound) { - esp_zb_zcl_on_off_cmd_t cmd_req; + esp_zb_zcl_on_off_cmd_t cmd_req = {0}; cmd_req.zcl_basic_cmd.src_endpoint = _endpoint; cmd_req.address_mode = ESP_ZB_APS_ADDR_MODE_DST_ADDR_ENDP_NOT_PRESENT; cmd_req.on_off_cmd_id = ESP_ZB_ZCL_CMD_ON_OFF_TOGGLE_ID; @@ -112,7 +112,7 @@ void ZigbeeColorDimmerSwitch::lightToggle() { void ZigbeeColorDimmerSwitch::lightToggle(uint16_t group_addr) { if (_is_bound) { - esp_zb_zcl_on_off_cmd_t cmd_req; + esp_zb_zcl_on_off_cmd_t cmd_req = {0}; 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; @@ -128,7 +128,7 @@ void ZigbeeColorDimmerSwitch::lightToggle(uint16_t group_addr) { void ZigbeeColorDimmerSwitch::lightToggle(uint8_t endpoint, uint16_t short_addr) { if (_is_bound) { - esp_zb_zcl_on_off_cmd_t cmd_req; + esp_zb_zcl_on_off_cmd_t cmd_req = {0}; 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; @@ -145,7 +145,7 @@ void ZigbeeColorDimmerSwitch::lightToggle(uint8_t endpoint, uint16_t short_addr) void ZigbeeColorDimmerSwitch::lightToggle(uint8_t endpoint, esp_zb_ieee_addr_t ieee_addr) { if (_is_bound) { - esp_zb_zcl_on_off_cmd_t cmd_req; + esp_zb_zcl_on_off_cmd_t cmd_req = {0}; 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; @@ -165,7 +165,7 @@ void ZigbeeColorDimmerSwitch::lightToggle(uint8_t endpoint, esp_zb_ieee_addr_t i void ZigbeeColorDimmerSwitch::lightOn() { if (_is_bound) { - esp_zb_zcl_on_off_cmd_t cmd_req; + esp_zb_zcl_on_off_cmd_t cmd_req = {0}; cmd_req.zcl_basic_cmd.src_endpoint = _endpoint; cmd_req.address_mode = ESP_ZB_APS_ADDR_MODE_DST_ADDR_ENDP_NOT_PRESENT; cmd_req.on_off_cmd_id = ESP_ZB_ZCL_CMD_ON_OFF_ON_ID; @@ -180,7 +180,7 @@ void ZigbeeColorDimmerSwitch::lightOn() { void ZigbeeColorDimmerSwitch::lightOn(uint16_t group_addr) { if (_is_bound) { - esp_zb_zcl_on_off_cmd_t cmd_req; + esp_zb_zcl_on_off_cmd_t cmd_req = {0}; 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; @@ -196,7 +196,7 @@ void ZigbeeColorDimmerSwitch::lightOn(uint16_t group_addr) { void ZigbeeColorDimmerSwitch::lightOn(uint8_t endpoint, uint16_t short_addr) { if (_is_bound) { - esp_zb_zcl_on_off_cmd_t cmd_req; + esp_zb_zcl_on_off_cmd_t cmd_req = {0}; 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; @@ -213,7 +213,7 @@ void ZigbeeColorDimmerSwitch::lightOn(uint8_t endpoint, uint16_t short_addr) { void ZigbeeColorDimmerSwitch::lightOn(uint8_t endpoint, esp_zb_ieee_addr_t ieee_addr) { if (_is_bound) { - esp_zb_zcl_on_off_cmd_t cmd_req; + esp_zb_zcl_on_off_cmd_t cmd_req = {0}; 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; @@ -233,7 +233,7 @@ void ZigbeeColorDimmerSwitch::lightOn(uint8_t endpoint, esp_zb_ieee_addr_t ieee_ void ZigbeeColorDimmerSwitch::lightOff() { if (_is_bound) { - esp_zb_zcl_on_off_cmd_t cmd_req; + esp_zb_zcl_on_off_cmd_t cmd_req = {0}; cmd_req.zcl_basic_cmd.src_endpoint = _endpoint; cmd_req.address_mode = ESP_ZB_APS_ADDR_MODE_DST_ADDR_ENDP_NOT_PRESENT; cmd_req.on_off_cmd_id = ESP_ZB_ZCL_CMD_ON_OFF_OFF_ID; @@ -248,7 +248,7 @@ void ZigbeeColorDimmerSwitch::lightOff() { void ZigbeeColorDimmerSwitch::lightOff(uint16_t group_addr) { if (_is_bound) { - esp_zb_zcl_on_off_cmd_t cmd_req; + esp_zb_zcl_on_off_cmd_t cmd_req = {0}; 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; @@ -264,7 +264,7 @@ void ZigbeeColorDimmerSwitch::lightOff(uint16_t group_addr) { void ZigbeeColorDimmerSwitch::lightOff(uint8_t endpoint, uint16_t short_addr) { if (_is_bound) { - esp_zb_zcl_on_off_cmd_t cmd_req; + esp_zb_zcl_on_off_cmd_t cmd_req = {0}; 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; @@ -281,7 +281,7 @@ void ZigbeeColorDimmerSwitch::lightOff(uint8_t endpoint, uint16_t short_addr) { void ZigbeeColorDimmerSwitch::lightOff(uint8_t endpoint, esp_zb_ieee_addr_t ieee_addr) { if (_is_bound) { - esp_zb_zcl_on_off_cmd_t cmd_req; + esp_zb_zcl_on_off_cmd_t cmd_req = {0}; 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; @@ -301,7 +301,7 @@ void ZigbeeColorDimmerSwitch::lightOff(uint8_t endpoint, esp_zb_ieee_addr_t ieee void ZigbeeColorDimmerSwitch::lightOffWithEffect(uint8_t effect_id, uint8_t effect_variant) { if (_is_bound) { - esp_zb_zcl_on_off_off_with_effect_cmd_t cmd_req; + esp_zb_zcl_on_off_off_with_effect_cmd_t cmd_req = {0}; cmd_req.zcl_basic_cmd.src_endpoint = _endpoint; cmd_req.address_mode = ESP_ZB_APS_ADDR_MODE_DST_ADDR_ENDP_NOT_PRESENT; cmd_req.effect_id = effect_id; @@ -317,7 +317,7 @@ void ZigbeeColorDimmerSwitch::lightOffWithEffect(uint8_t effect_id, uint8_t effe void ZigbeeColorDimmerSwitch::lightOnWithSceneRecall() { if (_is_bound) { - esp_zb_zcl_on_off_on_with_recall_global_scene_cmd_t cmd_req; + esp_zb_zcl_on_off_on_with_recall_global_scene_cmd_t cmd_req = {0}; cmd_req.zcl_basic_cmd.src_endpoint = _endpoint; cmd_req.address_mode = ESP_ZB_APS_ADDR_MODE_DST_ADDR_ENDP_NOT_PRESENT; log_v("Sending 'light on with scene recall' command"); @@ -331,7 +331,7 @@ void ZigbeeColorDimmerSwitch::lightOnWithSceneRecall() { void ZigbeeColorDimmerSwitch::lightOnWithTimedOff(uint8_t on_off_control, uint16_t time_on, uint16_t time_off) { if (_is_bound) { - esp_zb_zcl_on_off_on_with_timed_off_cmd_t cmd_req; + esp_zb_zcl_on_off_on_with_timed_off_cmd_t cmd_req = {0}; cmd_req.zcl_basic_cmd.src_endpoint = _endpoint; cmd_req.address_mode = ESP_ZB_APS_ADDR_MODE_DST_ADDR_ENDP_NOT_PRESENT; cmd_req.on_off_control = on_off_control; //TODO: Test how it works, then maybe change API @@ -348,7 +348,7 @@ void ZigbeeColorDimmerSwitch::lightOnWithTimedOff(uint8_t on_off_control, uint16 void ZigbeeColorDimmerSwitch::setLightLevel(uint8_t level) { if (_is_bound) { - esp_zb_zcl_move_to_level_cmd_t cmd_req; + esp_zb_zcl_move_to_level_cmd_t cmd_req = {0}; cmd_req.zcl_basic_cmd.src_endpoint = _endpoint; cmd_req.address_mode = ESP_ZB_APS_ADDR_MODE_DST_ADDR_ENDP_NOT_PRESENT; cmd_req.level = level; @@ -364,7 +364,7 @@ void ZigbeeColorDimmerSwitch::setLightLevel(uint8_t level) { void ZigbeeColorDimmerSwitch::setLightLevel(uint8_t level, uint16_t group_addr) { if (_is_bound) { - esp_zb_zcl_move_to_level_cmd_t cmd_req; + esp_zb_zcl_move_to_level_cmd_t cmd_req = {0}; 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; @@ -381,7 +381,7 @@ void ZigbeeColorDimmerSwitch::setLightLevel(uint8_t level, uint16_t group_addr) void ZigbeeColorDimmerSwitch::setLightLevel(uint8_t level, uint8_t endpoint, uint16_t short_addr) { if (_is_bound) { - esp_zb_zcl_move_to_level_cmd_t cmd_req; + esp_zb_zcl_move_to_level_cmd_t cmd_req = {0}; 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; @@ -399,7 +399,7 @@ void ZigbeeColorDimmerSwitch::setLightLevel(uint8_t level, uint8_t endpoint, uin void ZigbeeColorDimmerSwitch::setLightLevel(uint8_t level, uint8_t endpoint, esp_zb_ieee_addr_t ieee_addr) { if (_is_bound) { - esp_zb_zcl_move_to_level_cmd_t cmd_req; + esp_zb_zcl_move_to_level_cmd_t cmd_req = {0}; 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; @@ -422,7 +422,7 @@ void ZigbeeColorDimmerSwitch::setLightColor(uint8_t red, uint8_t green, uint8_t if (_is_bound) { espXyColor_t xy_color = espRgbToXYColor(red, green, blue); - esp_zb_zcl_color_move_to_color_cmd_t cmd_req; + esp_zb_zcl_color_move_to_color_cmd_t cmd_req = {0}; 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; @@ -441,7 +441,7 @@ void ZigbeeColorDimmerSwitch::setLightColor(uint8_t red, uint8_t green, uint8_t if (_is_bound) { espXyColor_t xy_color = espRgbToXYColor(red, green, blue); - esp_zb_zcl_color_move_to_color_cmd_t cmd_req; + esp_zb_zcl_color_move_to_color_cmd_t cmd_req = {0}; 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; @@ -461,7 +461,7 @@ void ZigbeeColorDimmerSwitch::setLightColor(uint8_t red, uint8_t green, uint8_t if (_is_bound) { espXyColor_t xy_color = espRgbToXYColor(red, green, blue); - esp_zb_zcl_color_move_to_color_cmd_t cmd_req; + esp_zb_zcl_color_move_to_color_cmd_t cmd_req = {0}; 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; @@ -482,7 +482,7 @@ void ZigbeeColorDimmerSwitch::setLightColor(uint8_t red, uint8_t green, uint8_t if (_is_bound) { espXyColor_t xy_color = espRgbToXYColor(red, green, blue); - esp_zb_zcl_color_move_to_color_cmd_t cmd_req; + esp_zb_zcl_color_move_to_color_cmd_t cmd_req = {0}; 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; diff --git a/libraries/Zigbee/src/ep/ZigbeeSwitch.cpp b/libraries/Zigbee/src/ep/ZigbeeSwitch.cpp index 38fd7d370fb..68e7a7430cc 100644 --- a/libraries/Zigbee/src/ep/ZigbeeSwitch.cpp +++ b/libraries/Zigbee/src/ep/ZigbeeSwitch.cpp @@ -52,7 +52,7 @@ void ZigbeeSwitch::findCb(esp_zb_zdp_status_t zdo_status, uint16_t addr, uint8_t ZigbeeSwitch *instance = static_cast(user_ctx); if (zdo_status == ESP_ZB_ZDP_STATUS_SUCCESS) { log_d("Found light endpoint"); - esp_zb_zdo_bind_req_param_t bind_req; + esp_zb_zdo_bind_req_param_t bind_req = {0}; zb_device_params_t *light = (zb_device_params_t *)malloc(sizeof(zb_device_params_t)); light->endpoint = endpoint; light->short_addr = addr; @@ -94,7 +94,7 @@ void ZigbeeSwitch::findEndpoint(esp_zb_zdo_match_desc_req_param_t *cmd_req) { // Methods to control the light void ZigbeeSwitch::lightToggle() { if (_is_bound) { - esp_zb_zcl_on_off_cmd_t cmd_req; + esp_zb_zcl_on_off_cmd_t cmd_req = {0}; cmd_req.zcl_basic_cmd.src_endpoint = _endpoint; cmd_req.address_mode = ESP_ZB_APS_ADDR_MODE_DST_ADDR_ENDP_NOT_PRESENT; cmd_req.on_off_cmd_id = ESP_ZB_ZCL_CMD_ON_OFF_TOGGLE_ID; @@ -109,7 +109,7 @@ void ZigbeeSwitch::lightToggle() { void ZigbeeSwitch::lightToggle(uint16_t group_addr) { if (_is_bound) { - esp_zb_zcl_on_off_cmd_t cmd_req; + esp_zb_zcl_on_off_cmd_t cmd_req = {0}; 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; @@ -125,7 +125,7 @@ void ZigbeeSwitch::lightToggle(uint16_t group_addr) { void ZigbeeSwitch::lightToggle(uint8_t endpoint, uint16_t short_addr) { if (_is_bound) { - esp_zb_zcl_on_off_cmd_t cmd_req; + esp_zb_zcl_on_off_cmd_t cmd_req = {0}; 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; @@ -142,7 +142,7 @@ void ZigbeeSwitch::lightToggle(uint8_t endpoint, uint16_t short_addr) { void ZigbeeSwitch::lightToggle(uint8_t endpoint, esp_zb_ieee_addr_t ieee_addr) { if (_is_bound) { - esp_zb_zcl_on_off_cmd_t cmd_req; + esp_zb_zcl_on_off_cmd_t cmd_req = {0}; 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; @@ -162,7 +162,7 @@ void ZigbeeSwitch::lightToggle(uint8_t endpoint, esp_zb_ieee_addr_t ieee_addr) { void ZigbeeSwitch::lightOn() { if (_is_bound) { - esp_zb_zcl_on_off_cmd_t cmd_req; + esp_zb_zcl_on_off_cmd_t cmd_req = {0}; cmd_req.zcl_basic_cmd.src_endpoint = _endpoint; cmd_req.address_mode = ESP_ZB_APS_ADDR_MODE_DST_ADDR_ENDP_NOT_PRESENT; cmd_req.on_off_cmd_id = ESP_ZB_ZCL_CMD_ON_OFF_ON_ID; @@ -177,7 +177,7 @@ void ZigbeeSwitch::lightOn() { void ZigbeeSwitch::lightOn(uint16_t group_addr) { if (_is_bound) { - esp_zb_zcl_on_off_cmd_t cmd_req; + esp_zb_zcl_on_off_cmd_t cmd_req = {0}; 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; @@ -193,7 +193,7 @@ void ZigbeeSwitch::lightOn(uint16_t group_addr) { void ZigbeeSwitch::lightOn(uint8_t endpoint, uint16_t short_addr) { if (_is_bound) { - esp_zb_zcl_on_off_cmd_t cmd_req; + esp_zb_zcl_on_off_cmd_t cmd_req = {0}; 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; @@ -210,7 +210,7 @@ void ZigbeeSwitch::lightOn(uint8_t endpoint, uint16_t short_addr) { void ZigbeeSwitch::lightOn(uint8_t endpoint, esp_zb_ieee_addr_t ieee_addr) { if (_is_bound) { - esp_zb_zcl_on_off_cmd_t cmd_req; + esp_zb_zcl_on_off_cmd_t cmd_req = {0}; 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; @@ -230,7 +230,7 @@ void ZigbeeSwitch::lightOn(uint8_t endpoint, esp_zb_ieee_addr_t ieee_addr) { void ZigbeeSwitch::lightOff() { if (_is_bound) { - esp_zb_zcl_on_off_cmd_t cmd_req; + esp_zb_zcl_on_off_cmd_t cmd_req = {0}; cmd_req.zcl_basic_cmd.src_endpoint = _endpoint; cmd_req.address_mode = ESP_ZB_APS_ADDR_MODE_DST_ADDR_ENDP_NOT_PRESENT; cmd_req.on_off_cmd_id = ESP_ZB_ZCL_CMD_ON_OFF_OFF_ID; @@ -245,7 +245,7 @@ void ZigbeeSwitch::lightOff() { void ZigbeeSwitch::lightOff(uint16_t group_addr) { if (_is_bound) { - esp_zb_zcl_on_off_cmd_t cmd_req; + esp_zb_zcl_on_off_cmd_t cmd_req = {0}; 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; @@ -261,7 +261,7 @@ void ZigbeeSwitch::lightOff(uint16_t group_addr) { void ZigbeeSwitch::lightOff(uint8_t endpoint, uint16_t short_addr) { if (_is_bound) { - esp_zb_zcl_on_off_cmd_t cmd_req; + esp_zb_zcl_on_off_cmd_t cmd_req = {0}; 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; @@ -278,7 +278,7 @@ void ZigbeeSwitch::lightOff(uint8_t endpoint, uint16_t short_addr) { void ZigbeeSwitch::lightOff(uint8_t endpoint, esp_zb_ieee_addr_t ieee_addr) { if (_is_bound) { - esp_zb_zcl_on_off_cmd_t cmd_req; + esp_zb_zcl_on_off_cmd_t cmd_req = {0}; 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; @@ -298,7 +298,7 @@ void ZigbeeSwitch::lightOff(uint8_t endpoint, esp_zb_ieee_addr_t ieee_addr) { void ZigbeeSwitch::lightOffWithEffect(uint8_t effect_id, uint8_t effect_variant) { if (_is_bound) { - esp_zb_zcl_on_off_off_with_effect_cmd_t cmd_req; + esp_zb_zcl_on_off_off_with_effect_cmd_t cmd_req = {0}; cmd_req.zcl_basic_cmd.src_endpoint = _endpoint; cmd_req.address_mode = ESP_ZB_APS_ADDR_MODE_DST_ADDR_ENDP_NOT_PRESENT; cmd_req.effect_id = effect_id; @@ -314,7 +314,7 @@ void ZigbeeSwitch::lightOffWithEffect(uint8_t effect_id, uint8_t effect_variant) void ZigbeeSwitch::lightOnWithSceneRecall() { if (_is_bound) { - esp_zb_zcl_on_off_on_with_recall_global_scene_cmd_t cmd_req; + esp_zb_zcl_on_off_on_with_recall_global_scene_cmd_t cmd_req = {0}; cmd_req.zcl_basic_cmd.src_endpoint = _endpoint; cmd_req.address_mode = ESP_ZB_APS_ADDR_MODE_DST_ADDR_ENDP_NOT_PRESENT; log_v("Sending 'light on with scene recall' command"); @@ -327,7 +327,7 @@ void ZigbeeSwitch::lightOnWithSceneRecall() { } void ZigbeeSwitch::lightOnWithTimedOff(uint8_t on_off_control, uint16_t time_on, uint16_t time_off) { if (_is_bound) { - esp_zb_zcl_on_off_on_with_timed_off_cmd_t cmd_req; + esp_zb_zcl_on_off_on_with_timed_off_cmd_t cmd_req = {0}; cmd_req.zcl_basic_cmd.src_endpoint = _endpoint; cmd_req.address_mode = ESP_ZB_APS_ADDR_MODE_DST_ADDR_ENDP_NOT_PRESENT; cmd_req.on_off_control = on_off_control; //TODO: Test how it works, then maybe change API diff --git a/libraries/Zigbee/src/ep/ZigbeeThermostat.cpp b/libraries/Zigbee/src/ep/ZigbeeThermostat.cpp index 633a60d0ff9..f3bfd7d981b 100644 --- a/libraries/Zigbee/src/ep/ZigbeeThermostat.cpp +++ b/libraries/Zigbee/src/ep/ZigbeeThermostat.cpp @@ -65,7 +65,7 @@ void ZigbeeThermostat::findCb(esp_zb_zdp_status_t zdo_status, uint16_t addr, uin ZigbeeThermostat *instance = static_cast(user_ctx); if (zdo_status == ESP_ZB_ZDP_STATUS_SUCCESS) { log_i("Found temperature sensor"); - esp_zb_zdo_bind_req_param_t bind_req; + esp_zb_zdo_bind_req_param_t bind_req = {0}; /* Store the information of the remote device */ zb_device_params_t *sensor = (zb_device_params_t *)malloc(sizeof(zb_device_params_t)); sensor->endpoint = endpoint; @@ -115,31 +115,38 @@ void ZigbeeThermostat::findEndpoint(esp_zb_zdo_match_desc_req_param_t *param) { esp_zb_zdo_match_cluster(param, ZigbeeThermostat::findCbWrapper, this); } -void ZigbeeThermostat::zbAttributeRead(uint16_t cluster_id, const esp_zb_zcl_attribute_t *attribute) { +void ZigbeeThermostat::zbAttributeRead(uint16_t cluster_id, const esp_zb_zcl_attribute_t *attribute, uint8_t src_endpoint, esp_zb_zcl_addr_t src_address) { static uint8_t read_config = 0; if (cluster_id == ESP_ZB_ZCL_CLUSTER_ID_TEMP_MEASUREMENT) { if (attribute->id == ESP_ZB_ZCL_ATTR_TEMP_MEASUREMENT_VALUE_ID && attribute->data.type == ESP_ZB_ZCL_ATTR_TYPE_S16) { int16_t value = attribute->data.value ? *(int16_t *)attribute->data.value : 0; - if (_on_temp_recieve) { - _on_temp_recieve(zb_s16_to_temperature(value)); + if (_on_temp_receive) { + _on_temp_receive(zb_s16_to_temperature(value)); + } + if (_on_temp_receive_with_source) { + _on_temp_receive_with_source(zb_s16_to_temperature(value), src_endpoint, src_address); } } if (attribute->id == ESP_ZB_ZCL_ATTR_TEMP_MEASUREMENT_MIN_VALUE_ID && attribute->data.type == ESP_ZB_ZCL_ATTR_TYPE_S16) { int16_t min_value = attribute->data.value ? *(int16_t *)attribute->data.value : 0; _min_temp = zb_s16_to_temperature(min_value); read_config++; + log_d("Received min temperature: %.2f°C from endpoint %d", _min_temp, src_endpoint); } if (attribute->id == ESP_ZB_ZCL_ATTR_TEMP_MEASUREMENT_MAX_VALUE_ID && attribute->data.type == ESP_ZB_ZCL_ATTR_TYPE_S16) { int16_t max_value = attribute->data.value ? *(int16_t *)attribute->data.value : 0; _max_temp = zb_s16_to_temperature(max_value); read_config++; + log_d("Received max temperature: %.2f°C from endpoint %d", _max_temp, src_endpoint); } if (attribute->id == ESP_ZB_ZCL_ATTR_TEMP_MEASUREMENT_TOLERANCE_ID && attribute->data.type == ESP_ZB_ZCL_ATTR_TYPE_U16) { uint16_t tolerance = attribute->data.value ? *(uint16_t *)attribute->data.value : 0; _tolerance = 1.0 * tolerance / 100; read_config++; + log_d("Received tolerance: %.2f°C from endpoint %d", _tolerance, src_endpoint); } if (read_config == 3) { + log_d("All config attributes processed"); read_config = 0; xSemaphoreGive(lock); } @@ -147,14 +154,14 @@ void ZigbeeThermostat::zbAttributeRead(uint16_t cluster_id, const esp_zb_zcl_att } void ZigbeeThermostat::getTemperature() { - /* Send "read attributes" command to the bound sensor */ - esp_zb_zcl_read_attr_cmd_t read_req; + /* Send "read attributes" command to all bound sensors */ + esp_zb_zcl_read_attr_cmd_t read_req = {0}; read_req.address_mode = ESP_ZB_APS_ADDR_MODE_DST_ADDR_ENDP_NOT_PRESENT; read_req.zcl_basic_cmd.src_endpoint = _endpoint; read_req.clusterID = ESP_ZB_ZCL_CLUSTER_ID_TEMP_MEASUREMENT; uint16_t attributes[] = {ESP_ZB_ZCL_ATTR_TEMP_MEASUREMENT_VALUE_ID}; - read_req.attr_number = ZB_ARRAY_LENTH(attributes); + read_req.attr_number = ZB_ARRAY_LENGHT(attributes); read_req.attr_field = attributes; log_i("Sending 'read temperature' command"); @@ -163,9 +170,68 @@ void ZigbeeThermostat::getTemperature() { esp_zb_lock_release(); } +void ZigbeeThermostat::getTemperature(uint16_t group_addr) { + /* Send "read attributes" command to the group */ + esp_zb_zcl_read_attr_cmd_t read_req = {0}; + read_req.address_mode = ESP_ZB_APS_ADDR_MODE_16_GROUP_ENDP_NOT_PRESENT; + read_req.zcl_basic_cmd.src_endpoint = _endpoint; + read_req.zcl_basic_cmd.dst_addr_u.addr_short = group_addr; + read_req.clusterID = ESP_ZB_ZCL_CLUSTER_ID_TEMP_MEASUREMENT; + + uint16_t attributes[] = {ESP_ZB_ZCL_ATTR_TEMP_MEASUREMENT_VALUE_ID}; + read_req.attr_number = ZB_ARRAY_LENGHT(attributes); + read_req.attr_field = attributes; + + log_i("Sending 'read temperature' command to group address 0x%x", group_addr); + esp_zb_lock_acquire(portMAX_DELAY); + esp_zb_zcl_read_attr_cmd_req(&read_req); + esp_zb_lock_release(); +} + +void ZigbeeThermostat::getTemperature(uint8_t endpoint, uint16_t short_addr) { + /* Send "read attributes" command to specific endpoint */ + esp_zb_zcl_read_attr_cmd_t read_req = {0}; + read_req.address_mode = ESP_ZB_APS_ADDR_MODE_16_ENDP_PRESENT; + read_req.zcl_basic_cmd.src_endpoint = _endpoint; + read_req.zcl_basic_cmd.dst_endpoint = endpoint; + read_req.zcl_basic_cmd.dst_addr_u.addr_short = short_addr; + read_req.clusterID = ESP_ZB_ZCL_CLUSTER_ID_TEMP_MEASUREMENT; + + uint16_t attributes[] = {ESP_ZB_ZCL_ATTR_TEMP_MEASUREMENT_VALUE_ID}; + read_req.attr_number = ZB_ARRAY_LENGHT(attributes); + read_req.attr_field = attributes; + + log_i("Sending 'read temperature' command to endpoint %d, address 0x%x", endpoint, short_addr); + esp_zb_lock_acquire(portMAX_DELAY); + esp_zb_zcl_read_attr_cmd_req(&read_req); + esp_zb_lock_release(); +} + +void ZigbeeThermostat::getTemperature(uint8_t endpoint, esp_zb_ieee_addr_t ieee_addr) { + /* Send "read attributes" command to specific endpoint */ + esp_zb_zcl_read_attr_cmd_t read_req = {0}; + read_req.address_mode = ESP_ZB_APS_ADDR_MODE_64_ENDP_PRESENT; + read_req.zcl_basic_cmd.src_endpoint = _endpoint; + read_req.zcl_basic_cmd.dst_endpoint = endpoint; + read_req.clusterID = ESP_ZB_ZCL_CLUSTER_ID_TEMP_MEASUREMENT; + 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_TEMP_MEASUREMENT_VALUE_ID}; + read_req.attr_number = ZB_ARRAY_LENGHT(attributes); + read_req.attr_field = attributes; + + log_i( + "Sending 'read temperature' command to endpoint %d, ieee address %02x:%02x:%02x:%02x:%02x:%02x:%02x:%02x", endpoint, ieee_addr[7], ieee_addr[6], + ieee_addr[5], ieee_addr[4], ieee_addr[3], ieee_addr[2], ieee_addr[1], ieee_addr[0] + ); + esp_zb_lock_acquire(portMAX_DELAY); + esp_zb_zcl_read_attr_cmd_req(&read_req); + esp_zb_lock_release(); +} + void ZigbeeThermostat::getSensorSettings() { - /* Send "read attributes" command to the bound sensor */ - esp_zb_zcl_read_attr_cmd_t read_req; + /* Send "read attributes" command to all bound sensors */ + esp_zb_zcl_read_attr_cmd_t read_req = {0}; read_req.address_mode = ESP_ZB_APS_ADDR_MODE_DST_ADDR_ENDP_NOT_PRESENT; read_req.zcl_basic_cmd.src_endpoint = _endpoint; read_req.clusterID = ESP_ZB_ZCL_CLUSTER_ID_TEMP_MEASUREMENT; @@ -173,10 +239,102 @@ void ZigbeeThermostat::getSensorSettings() { uint16_t attributes[] = { ESP_ZB_ZCL_ATTR_TEMP_MEASUREMENT_MIN_VALUE_ID, ESP_ZB_ZCL_ATTR_TEMP_MEASUREMENT_MAX_VALUE_ID, ESP_ZB_ZCL_ATTR_TEMP_MEASUREMENT_TOLERANCE_ID }; - read_req.attr_number = ZB_ARRAY_LENTH(attributes); + read_req.attr_number = ZB_ARRAY_LENGHT(attributes); read_req.attr_field = attributes; - log_i("Sending 'read temperature' command"); + log_i("Sending 'read sensor settings' command"); + esp_zb_lock_acquire(portMAX_DELAY); + esp_zb_zcl_read_attr_cmd_req(&read_req); + esp_zb_lock_release(); + + //Take semaphore to wait for response of all attributes + if (xSemaphoreTake(lock, ZB_CMD_TIMEOUT) != pdTRUE) { + log_e("Error while reading attributes"); + return; + } else { + //Call the callback function when all attributes are read + _on_config_receive(_min_temp, _max_temp, _tolerance); + } +} + +void ZigbeeThermostat::getSensorSettings(uint16_t group_addr) { + /* Send "read attributes" command to the group */ + esp_zb_zcl_read_attr_cmd_t read_req = {0}; + read_req.address_mode = ESP_ZB_APS_ADDR_MODE_16_GROUP_ENDP_NOT_PRESENT; + read_req.zcl_basic_cmd.src_endpoint = _endpoint; + read_req.zcl_basic_cmd.dst_addr_u.addr_short = group_addr; + read_req.clusterID = ESP_ZB_ZCL_CLUSTER_ID_TEMP_MEASUREMENT; + + uint16_t attributes[] = { + ESP_ZB_ZCL_ATTR_TEMP_MEASUREMENT_MIN_VALUE_ID, ESP_ZB_ZCL_ATTR_TEMP_MEASUREMENT_MAX_VALUE_ID, ESP_ZB_ZCL_ATTR_TEMP_MEASUREMENT_TOLERANCE_ID + }; + read_req.attr_number = ZB_ARRAY_LENGHT(attributes); + read_req.attr_field = attributes; + + log_i("Sending 'read sensor settings' command to group address 0x%x", group_addr); + esp_zb_lock_acquire(portMAX_DELAY); + esp_zb_zcl_read_attr_cmd_req(&read_req); + esp_zb_lock_release(); + + //Take semaphore to wait for response of all attributes + if (xSemaphoreTake(lock, ZB_CMD_TIMEOUT) != pdTRUE) { + log_e("Error while reading attributes"); + return; + } else { + //Call the callback function when all attributes are read + _on_config_receive(_min_temp, _max_temp, _tolerance); + } +} + +void ZigbeeThermostat::getSensorSettings(uint8_t endpoint, uint16_t short_addr) { + /* Send "read attributes" command to specific endpoint */ + esp_zb_zcl_read_attr_cmd_t read_req = {0}; + read_req.address_mode = ESP_ZB_APS_ADDR_MODE_16_ENDP_PRESENT; + read_req.zcl_basic_cmd.src_endpoint = _endpoint; + read_req.zcl_basic_cmd.dst_endpoint = endpoint; + read_req.zcl_basic_cmd.dst_addr_u.addr_short = short_addr; + read_req.clusterID = ESP_ZB_ZCL_CLUSTER_ID_TEMP_MEASUREMENT; + + uint16_t attributes[] = { + ESP_ZB_ZCL_ATTR_TEMP_MEASUREMENT_MIN_VALUE_ID, ESP_ZB_ZCL_ATTR_TEMP_MEASUREMENT_MAX_VALUE_ID, ESP_ZB_ZCL_ATTR_TEMP_MEASUREMENT_TOLERANCE_ID + }; + read_req.attr_number = ZB_ARRAY_LENGHT(attributes); + read_req.attr_field = attributes; + + log_i("Sending 'read sensor settings' command to endpoint %d, address 0x%x", endpoint, short_addr); + esp_zb_lock_acquire(portMAX_DELAY); + esp_zb_zcl_read_attr_cmd_req(&read_req); + esp_zb_lock_release(); + + //Take semaphore to wait for response of all attributes + if (xSemaphoreTake(lock, ZB_CMD_TIMEOUT) != pdTRUE) { + log_e("Error while reading attributes"); + return; + } else { + //Call the callback function when all attributes are read + _on_config_receive(_min_temp, _max_temp, _tolerance); + } +} + +void ZigbeeThermostat::getSensorSettings(uint8_t endpoint, esp_zb_ieee_addr_t ieee_addr) { + /* Send "read attributes" command to specific endpoint */ + esp_zb_zcl_read_attr_cmd_t read_req = {0}; + read_req.address_mode = ESP_ZB_APS_ADDR_MODE_64_ENDP_PRESENT; + read_req.zcl_basic_cmd.src_endpoint = _endpoint; + read_req.zcl_basic_cmd.dst_endpoint = endpoint; + read_req.clusterID = ESP_ZB_ZCL_CLUSTER_ID_TEMP_MEASUREMENT; + 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_TEMP_MEASUREMENT_MIN_VALUE_ID, ESP_ZB_ZCL_ATTR_TEMP_MEASUREMENT_MAX_VALUE_ID, ESP_ZB_ZCL_ATTR_TEMP_MEASUREMENT_TOLERANCE_ID + }; + read_req.attr_number = ZB_ARRAY_LENGHT(attributes); + read_req.attr_field = attributes; + + log_i( + "Sending 'read sensor settings' command to endpoint %d, ieee address %02x:%02x:%02x:%02x:%02x:%02x:%02x:%02x", endpoint, ieee_addr[7], ieee_addr[6], + ieee_addr[5], ieee_addr[4], ieee_addr[3], ieee_addr[2], ieee_addr[1], ieee_addr[0] + ); esp_zb_lock_acquire(portMAX_DELAY); esp_zb_zcl_read_attr_cmd_req(&read_req); esp_zb_lock_release(); @@ -187,13 +345,13 @@ void ZigbeeThermostat::getSensorSettings() { return; } else { //Call the callback function when all attributes are read - _on_config_recieve(_min_temp, _max_temp, _tolerance); + _on_config_receive(_min_temp, _max_temp, _tolerance); } } void ZigbeeThermostat::setTemperatureReporting(uint16_t min_interval, uint16_t max_interval, float delta) { - /* Send "configure report attribute" command to the bound sensor */ - esp_zb_zcl_config_report_cmd_t report_cmd; + /* Send "configure report attribute" command to all bound sensors */ + esp_zb_zcl_config_report_cmd_t report_cmd = {0}; report_cmd.address_mode = ESP_ZB_APS_ADDR_MODE_DST_ADDR_ENDP_NOT_PRESENT; report_cmd.zcl_basic_cmd.src_endpoint = _endpoint; report_cmd.clusterID = ESP_ZB_ZCL_CLUSTER_ID_TEMP_MEASUREMENT; @@ -206,10 +364,10 @@ void ZigbeeThermostat::setTemperatureReporting(uint16_t min_interval, uint16_t m .attrType = ESP_ZB_ZCL_ATTR_TYPE_S16, .min_interval = min_interval, .max_interval = max_interval, - .reportable_change = &report_change, + .reportable_change = (void *)&report_change, }, }; - report_cmd.record_number = ZB_ARRAY_LENTH(records); + report_cmd.record_number = ZB_ARRAY_LENGHT(records); report_cmd.record_field = records; log_i("Sending 'configure reporting' command"); @@ -218,4 +376,93 @@ void ZigbeeThermostat::setTemperatureReporting(uint16_t min_interval, uint16_t m esp_zb_lock_release(); } +void ZigbeeThermostat::setTemperatureReporting(uint16_t group_addr, uint16_t min_interval, uint16_t max_interval, float delta) { + /* Send "configure report attribute" command to the group */ + esp_zb_zcl_config_report_cmd_t report_cmd = {0}; + report_cmd.address_mode = ESP_ZB_APS_ADDR_MODE_16_GROUP_ENDP_NOT_PRESENT; + report_cmd.zcl_basic_cmd.src_endpoint = _endpoint; + report_cmd.zcl_basic_cmd.dst_addr_u.addr_short = group_addr; + report_cmd.clusterID = ESP_ZB_ZCL_CLUSTER_ID_TEMP_MEASUREMENT; + + int16_t report_change = (int16_t)delta * 100; + esp_zb_zcl_config_report_record_t records[] = { + { + .direction = ESP_ZB_ZCL_REPORT_DIRECTION_SEND, + .attributeID = ESP_ZB_ZCL_ATTR_TEMP_MEASUREMENT_VALUE_ID, + .attrType = ESP_ZB_ZCL_ATTR_TYPE_S16, + .min_interval = min_interval, + .max_interval = max_interval, + .reportable_change = (void *)&report_change, + }, + }; + report_cmd.record_number = ZB_ARRAY_LENGHT(records); + report_cmd.record_field = records; + + log_i("Sending 'configure reporting' command to group address 0x%x", group_addr); + esp_zb_lock_acquire(portMAX_DELAY); + esp_zb_zcl_config_report_cmd_req(&report_cmd); + esp_zb_lock_release(); +} + +void ZigbeeThermostat::setTemperatureReporting(uint8_t endpoint, uint16_t short_addr, uint16_t min_interval, uint16_t max_interval, float delta) { + /* Send "configure report attribute" command to specific endpoint */ + esp_zb_zcl_config_report_cmd_t report_cmd = {0}; + report_cmd.address_mode = ESP_ZB_APS_ADDR_MODE_16_ENDP_PRESENT; + report_cmd.zcl_basic_cmd.src_endpoint = _endpoint; + report_cmd.zcl_basic_cmd.dst_endpoint = endpoint; + report_cmd.zcl_basic_cmd.dst_addr_u.addr_short = short_addr; + report_cmd.clusterID = ESP_ZB_ZCL_CLUSTER_ID_TEMP_MEASUREMENT; + + int16_t report_change = (int16_t)delta * 100; + esp_zb_zcl_config_report_record_t records[] = { + { + .direction = ESP_ZB_ZCL_REPORT_DIRECTION_SEND, + .attributeID = ESP_ZB_ZCL_ATTR_TEMP_MEASUREMENT_VALUE_ID, + .attrType = ESP_ZB_ZCL_ATTR_TYPE_S16, + .min_interval = min_interval, + .max_interval = max_interval, + .reportable_change = (void *)&report_change, + }, + }; + report_cmd.record_number = ZB_ARRAY_LENGHT(records); + report_cmd.record_field = records; + + log_i("Sending 'configure reporting' command to endpoint %d, address 0x%x", endpoint, short_addr); + esp_zb_lock_acquire(portMAX_DELAY); + esp_zb_zcl_config_report_cmd_req(&report_cmd); + esp_zb_lock_release(); +} + +void ZigbeeThermostat::setTemperatureReporting(uint8_t endpoint, esp_zb_ieee_addr_t ieee_addr, uint16_t min_interval, uint16_t max_interval, float delta) { + /* Send "configure report attribute" command to specific endpoint */ + esp_zb_zcl_config_report_cmd_t report_cmd = {0}; + report_cmd.address_mode = ESP_ZB_APS_ADDR_MODE_64_ENDP_PRESENT; + report_cmd.zcl_basic_cmd.src_endpoint = _endpoint; + report_cmd.zcl_basic_cmd.dst_endpoint = endpoint; + report_cmd.clusterID = ESP_ZB_ZCL_CLUSTER_ID_TEMP_MEASUREMENT; + memcpy(report_cmd.zcl_basic_cmd.dst_addr_u.addr_long, ieee_addr, sizeof(esp_zb_ieee_addr_t)); + + int16_t report_change = (int16_t)delta * 100; + esp_zb_zcl_config_report_record_t records[] = { + { + .direction = ESP_ZB_ZCL_REPORT_DIRECTION_SEND, + .attributeID = ESP_ZB_ZCL_ATTR_TEMP_MEASUREMENT_VALUE_ID, + .attrType = ESP_ZB_ZCL_ATTR_TYPE_S16, + .min_interval = min_interval, + .max_interval = max_interval, + .reportable_change = (void *)&report_change, + }, + }; + report_cmd.record_number = ZB_ARRAY_LENGHT(records); + report_cmd.record_field = records; + + log_i( + "Sending 'configure reporting' command to endpoint %d, ieee address %02x:%02x:%02x:%02x:%02x:%02x:%02x:%02x", endpoint, ieee_addr[7], ieee_addr[6], + ieee_addr[5], ieee_addr[4], ieee_addr[3], ieee_addr[2], ieee_addr[1], ieee_addr[0] + ); + esp_zb_lock_acquire(portMAX_DELAY); + esp_zb_zcl_config_report_cmd_req(&report_cmd); + esp_zb_lock_release(); +} + #endif // CONFIG_ZB_ENABLED diff --git a/libraries/Zigbee/src/ep/ZigbeeThermostat.h b/libraries/Zigbee/src/ep/ZigbeeThermostat.h index acdcd68e512..c10a9d7f974 100644 --- a/libraries/Zigbee/src/ep/ZigbeeThermostat.h +++ b/libraries/Zigbee/src/ep/ZigbeeThermostat.h @@ -34,24 +34,39 @@ class ZigbeeThermostat : public ZigbeeEP { ZigbeeThermostat(uint8_t endpoint); ~ZigbeeThermostat() {} - void onTempRecieve(void (*callback)(float)) { - _on_temp_recieve = callback; + void onTempReceive(void (*callback)(float)) { + _on_temp_receive = callback; } - void onConfigRecieve(void (*callback)(float, float, float)) { - _on_config_recieve = callback; + void onTempReceiveWithSource(void (*callback)(float, uint8_t, esp_zb_zcl_addr_t)) { + _on_temp_receive_with_source = callback; + } + void onConfigReceive(void (*callback)(float, float, float)) { + _on_config_receive = callback; } void getTemperature(); + void getTemperature(uint16_t group_addr); + void getTemperature(uint8_t endpoint, uint16_t short_addr); + void getTemperature(uint8_t endpoint, esp_zb_ieee_addr_t ieee_addr); + void getSensorSettings(); + void getSensorSettings(uint16_t group_addr); + void getSensorSettings(uint8_t endpoint, uint16_t short_addr); + void getSensorSettings(uint8_t endpoint, esp_zb_ieee_addr_t ieee_addr); + void setTemperatureReporting(uint16_t min_interval, uint16_t max_interval, float delta); + void setTemperatureReporting(uint16_t group_addr, uint16_t min_interval, uint16_t max_interval, float delta); + void setTemperatureReporting(uint8_t endpoint, uint16_t short_addr, uint16_t min_interval, uint16_t max_interval, float delta); + void setTemperatureReporting(uint8_t endpoint, esp_zb_ieee_addr_t ieee_addr, uint16_t min_interval, uint16_t max_interval, float delta); private: // save instance of the class in order to use it in static functions static ZigbeeThermostat *_instance; zb_device_params_t *_device; - void (*_on_temp_recieve)(float); - void (*_on_config_recieve)(float, float, float); + void (*_on_temp_receive)(float); + void (*_on_temp_receive_with_source)(float, uint8_t, esp_zb_zcl_addr_t); + void (*_on_config_receive)(float, float, float); float _min_temp; float _max_temp; float _tolerance; @@ -62,7 +77,7 @@ class ZigbeeThermostat : public ZigbeeEP { static void bindCbWrapper(esp_zb_zdp_status_t zdo_status, void *user_ctx); static void findCbWrapper(esp_zb_zdp_status_t zdo_status, uint16_t addr, uint8_t endpoint, void *user_ctx); - void zbAttributeRead(uint16_t cluster_id, const esp_zb_zcl_attribute_t *attribute) override; + void zbAttributeRead(uint16_t cluster_id, const esp_zb_zcl_attribute_t *attribute, uint8_t src_endpoint, esp_zb_zcl_addr_t src_address) override; }; #endif // CONFIG_ZB_ENABLED From e3018b67191ec621cbd28b872f99c259de83a7f4 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jan=20Proch=C3=A1zka?= <90197375+P-R-O-C-H-Y@users.noreply.github.com> Date: Wed, 4 Jun 2025 16:40:52 +0200 Subject: [PATCH 034/173] feat(zigbee): Add method to set/get/report analog output (#11431) * feat(zigbee): Add methot to set,get,report analog output * fix(ci): Update json file for example * fix(zigbee): Add missing keywords * ci(pre-commit): Apply automatic fixes --------- Co-authored-by: pre-commit-ci-lite[bot] <117423508+pre-commit-ci-lite[bot]@users.noreply.github.com> --- .../Zigbee_Analog_Input_Output.ino | 12 +++-- .../Zigbee_Analog_Input_Output/ci.json | 3 +- libraries/Zigbee/keywords.txt | 11 ++++- libraries/Zigbee/src/ep/ZigbeeAnalog.cpp | 49 +++++++++++++++++-- libraries/Zigbee/src/ep/ZigbeeAnalog.h | 14 ++++-- 5 files changed, 74 insertions(+), 15 deletions(-) diff --git a/libraries/Zigbee/examples/Zigbee_Analog_Input_Output/Zigbee_Analog_Input_Output.ino b/libraries/Zigbee/examples/Zigbee_Analog_Input_Output/Zigbee_Analog_Input_Output.ino index db8a62b091e..59c4b514db1 100644 --- a/libraries/Zigbee/examples/Zigbee_Analog_Input_Output/Zigbee_Analog_Input_Output.ino +++ b/libraries/Zigbee/examples/Zigbee_Analog_Input_Output/Zigbee_Analog_Input_Output.ino @@ -26,8 +26,8 @@ * Modified by Pat Clay */ -#ifndef ZIGBEE_MODE_ED -#error "Zigbee end device mode is not selected in Tools->Zigbee mode" +#ifndef ZIGBEE_MODE_ZCZR +#error "Zigbee coordinator/router device mode is not selected in Tools->Zigbee mode" #endif #include "Zigbee.h" @@ -70,6 +70,7 @@ void setup() { zbAnalogDevice.addAnalogOutput(); zbAnalogDevice.setAnalogOutputApplication(ESP_ZB_ZCL_AI_RPM_OTHER); zbAnalogDevice.setAnalogOutputDescription("Fan Speed (RPM)"); + zbAnalogDevice.setAnalogOutputResolution(1); // If analog output cluster is added, set callback function for analog output change zbAnalogDevice.onAnalogOutputChange(onAnalogOutputChange); @@ -99,8 +100,8 @@ void setup() { Zigbee.addEndpoint(&zbAnalogPercent); Serial.println("Starting Zigbee..."); - // When all EPs are registered, start Zigbee in End Device mode - if (!Zigbee.begin()) { + // When all EPs are registered, start Zigbee in Router Device mode + if (!Zigbee.begin(ZIGBEE_ROUTER)) { Serial.println("Zigbee failed to start!"); Serial.println("Rebooting..."); ESP.restart(); @@ -151,6 +152,9 @@ void loop() { Zigbee.factoryReset(); } } + // For demonstration purposes, increment the analog output value by 100 + zbAnalogDevice.setAnalogOutput(zbAnalogDevice.getAnalogOutput() + 100); + zbAnalogDevice.reportAnalogOutput(); } delay(100); } diff --git a/libraries/Zigbee/examples/Zigbee_Analog_Input_Output/ci.json b/libraries/Zigbee/examples/Zigbee_Analog_Input_Output/ci.json index ceacc367801..15d6190e4ae 100644 --- a/libraries/Zigbee/examples/Zigbee_Analog_Input_Output/ci.json +++ b/libraries/Zigbee/examples/Zigbee_Analog_Input_Output/ci.json @@ -1,7 +1,6 @@ { - "fqbn_append": "PartitionScheme=zigbee,ZigbeeMode=ed", + "fqbn_append": "PartitionScheme=zigbee_zczr,ZigbeeMode=zczr", "requires": [ - "CONFIG_SOC_IEEE802154_SUPPORTED=y", "CONFIG_ZB_ENABLED=y" ] } diff --git a/libraries/Zigbee/keywords.txt b/libraries/Zigbee/keywords.txt index 5ce1e8f6f51..556b6408ea2 100644 --- a/libraries/Zigbee/keywords.txt +++ b/libraries/Zigbee/keywords.txt @@ -143,14 +143,21 @@ setSensorType KEYWORD2 setCarbonDioxide KEYWORD2 # ZigbeeAnalog -addAnalogValue KEYWORD2 addAnalogInput KEYWORD2 addAnalogOutput KEYWORD2 onAnalogOutputChange KEYWORD2 -setAnalogValue KEYWORD2 setAnalogInput KEYWORD2 +setAnalogOutput KEYWORD2 +getAnalogOutput KEYWORD2 reportAnalogInput KEYWORD2 +reportAnalogOutput KEYWORD2 setAnalogInputReporting KEYWORD2 +setAnalogInputApplication KEYWORD2 +setAnalogInputDescription KEYWORD2 +setAnalogInputResolution KEYWORD2 +setAnalogOutputApplication KEYWORD2 +setAnalogOutputDescription KEYWORD2 +setAnalogOutputResolution KEYWORD2 # ZigbeeCarbonDioxideSensor setCarbonDioxide KEYWORD2 diff --git a/libraries/Zigbee/src/ep/ZigbeeAnalog.cpp b/libraries/Zigbee/src/ep/ZigbeeAnalog.cpp index 6e073c345bc..893a9854ecc 100644 --- a/libraries/Zigbee/src/ep/ZigbeeAnalog.cpp +++ b/libraries/Zigbee/src/ep/ZigbeeAnalog.cpp @@ -129,8 +129,8 @@ bool ZigbeeAnalog::setAnalogOutputApplication(uint32_t application_type) { void ZigbeeAnalog::zbAttributeSet(const esp_zb_zcl_set_attr_value_message_t *message) { if (message->info.cluster == ESP_ZB_ZCL_CLUSTER_ID_ANALOG_OUTPUT) { if (message->attribute.id == ESP_ZB_ZCL_ATTR_ANALOG_OUTPUT_PRESENT_VALUE_ID && message->attribute.data.type == ESP_ZB_ZCL_ATTR_TYPE_SINGLE) { - float analog_output = *(float *)message->attribute.data.value; - analogOutputChanged(analog_output); + _output_state = *(float *)message->attribute.data.value; + analogOutputChanged(); } else { log_w("Received message ignored. Attribute ID: %d not supported for Analog Output", message->attribute.id); } @@ -139,9 +139,9 @@ void ZigbeeAnalog::zbAttributeSet(const esp_zb_zcl_set_attr_value_message_t *mes } } -void ZigbeeAnalog::analogOutputChanged(float analog_output) { +void ZigbeeAnalog::analogOutputChanged() { if (_on_analog_output_change) { - _on_analog_output_change(analog_output); + _on_analog_output_change(_output_state); } else { log_w("No callback function set for analog output change"); } @@ -166,6 +166,26 @@ bool ZigbeeAnalog::setAnalogInput(float analog) { return true; } +bool ZigbeeAnalog::setAnalogOutput(float analog) { + esp_zb_zcl_status_t ret = ESP_ZB_ZCL_STATUS_SUCCESS; + _output_state = analog; + analogOutputChanged(); + + log_v("Updating analog output to %.2f", analog); + /* Update analog output */ + esp_zb_lock_acquire(portMAX_DELAY); + ret = esp_zb_zcl_set_attribute_val( + _endpoint, ESP_ZB_ZCL_CLUSTER_ID_ANALOG_OUTPUT, ESP_ZB_ZCL_CLUSTER_SERVER_ROLE, ESP_ZB_ZCL_ATTR_ANALOG_OUTPUT_PRESENT_VALUE_ID, &_output_state, false + ); + esp_zb_lock_release(); + + if (ret != ESP_ZB_ZCL_STATUS_SUCCESS) { + log_e("Failed to set analog output: 0x%x: %s", ret, esp_zb_zcl_status_to_name(ret)); + return false; + } + return true; +} + bool ZigbeeAnalog::reportAnalogInput() { /* Send report attributes command */ esp_zb_zcl_report_attr_cmd_t report_attr_cmd; @@ -187,6 +207,27 @@ bool ZigbeeAnalog::reportAnalogInput() { return true; } +bool ZigbeeAnalog::reportAnalogOutput() { + /* 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_ANALOG_OUTPUT_PRESENT_VALUE_ID; + report_attr_cmd.direction = ESP_ZB_ZCL_CMD_DIRECTION_TO_CLI; + report_attr_cmd.clusterID = ESP_ZB_ZCL_CLUSTER_ID_ANALOG_OUTPUT; + report_attr_cmd.zcl_basic_cmd.src_endpoint = _endpoint; + report_attr_cmd.manuf_code = ESP_ZB_ZCL_ATTR_NON_MANUFACTURER_SPECIFIC; + + esp_zb_lock_acquire(portMAX_DELAY); + esp_err_t ret = esp_zb_zcl_report_attr_cmd_req(&report_attr_cmd); + esp_zb_lock_release(); + if (ret != ESP_OK) { + log_e("Failed to send Analog Output report: 0x%x: %s", ret, esp_err_to_name(ret)); + return false; + } + log_v("Analog Output report sent"); + return true; +} + bool ZigbeeAnalog::setAnalogInputReporting(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)); diff --git a/libraries/Zigbee/src/ep/ZigbeeAnalog.h b/libraries/Zigbee/src/ep/ZigbeeAnalog.h index e0c3fc11c5a..bbc5f6d6fc2 100644 --- a/libraries/Zigbee/src/ep/ZigbeeAnalog.h +++ b/libraries/Zigbee/src/ep/ZigbeeAnalog.h @@ -46,11 +46,18 @@ class ZigbeeAnalog : public ZigbeeEP { _on_analog_output_change = callback; } - // Set the analog input value + // Set the Analog Input/Output value bool setAnalogInput(float analog); + bool setAnalogOutput(float analog); - // Report Analog Input value + // Get the Analog Output value + float getAnalogOutput() { + return _output_state; + } + + // Report Analog Input/Output bool reportAnalogInput(); + bool reportAnalogOutput(); // Set reporting for Analog Input bool setAnalogInputReporting(uint16_t min_interval, uint16_t max_interval, float delta); @@ -59,9 +66,10 @@ class ZigbeeAnalog : public ZigbeeEP { void zbAttributeSet(const esp_zb_zcl_set_attr_value_message_t *message) override; void (*_on_analog_output_change)(float); - void analogOutputChanged(float analog_output); + void analogOutputChanged(); uint8_t _analog_clusters; + float _output_state; }; #endif // CONFIG_ZB_ENABLED From 0ab9a0fe6fb450f527cb277d09b70065905cf807 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jan=20Proch=C3=A1zka?= <90197375+P-R-O-C-H-Y@users.noreply.github.com> Date: Wed, 4 Jun 2025 17:48:48 +0200 Subject: [PATCH 035/173] feat(zigbee): Update to esp-zigbee-sdk 1.6.5 and fix ci.json files (#11436) * feat(zigbee): Update esp-zigbee-sdk and fix ci.json files * fix(ci): Check if LED_BUILTIN exist * ci(pre-commit): Apply automatic fixes --------- Co-authored-by: pre-commit-ci-lite[bot] <117423508+pre-commit-ci-lite[bot]@users.noreply.github.com> --- idf_component.yml | 4 ++-- .../Zigbee/examples/Zigbee_On_Off_MultiSwitch/ci.json | 2 +- .../examples/Zigbee_Power_Outlet/Zigbee_Power_Outlet.ino | 8 +++++++- libraries/Zigbee/examples/Zigbee_Power_Outlet/ci.json | 1 - 4 files changed, 10 insertions(+), 5 deletions(-) diff --git a/idf_component.yml b/idf_component.yml index dca97dc1655..766704176b0 100644 --- a/idf_component.yml +++ b/idf_component.yml @@ -54,12 +54,12 @@ dependencies: espressif/esp_modem: version: "^1.1.0" espressif/esp-zboss-lib: - version: "==1.6.3" + version: "==1.6.4" # compatible with esp-zigbee-lib 1.6.5 require: public rules: - if: "target not in [esp32c2, esp32p4]" espressif/esp-zigbee-lib: - version: "==1.6.3" + version: "==1.6.5" require: public rules: - if: "target not in [esp32c2, esp32p4]" diff --git a/libraries/Zigbee/examples/Zigbee_On_Off_MultiSwitch/ci.json b/libraries/Zigbee/examples/Zigbee_On_Off_MultiSwitch/ci.json index e79a477da11..15d6190e4ae 100644 --- a/libraries/Zigbee/examples/Zigbee_On_Off_MultiSwitch/ci.json +++ b/libraries/Zigbee/examples/Zigbee_On_Off_MultiSwitch/ci.json @@ -1,6 +1,6 @@ { "fqbn_append": "PartitionScheme=zigbee_zczr,ZigbeeMode=zczr", "requires": [ - "CONFIG_SOC_IEEE802154_SUPPORTED=y" + "CONFIG_ZB_ENABLED=y" ] } diff --git a/libraries/Zigbee/examples/Zigbee_Power_Outlet/Zigbee_Power_Outlet.ino b/libraries/Zigbee/examples/Zigbee_Power_Outlet/Zigbee_Power_Outlet.ino index d9fe1b7aefe..ba1474d6455 100644 --- a/libraries/Zigbee/examples/Zigbee_Power_Outlet/Zigbee_Power_Outlet.ino +++ b/libraries/Zigbee/examples/Zigbee_Power_Outlet/Zigbee_Power_Outlet.ino @@ -34,7 +34,13 @@ /* Zigbee power outlet configuration */ #define ZIGBEE_OUTLET_ENDPOINT 1 -uint8_t led = RGB_BUILTIN; + +#ifdef LED_BUILTIN // Use built-in LED if defined for the board +uint8_t led = LED_BUILTIN; +#else +uint8_t led = 2; // Use custom LED pin +#endif + uint8_t button = BOOT_PIN; ZigbeePowerOutlet zbOutlet = ZigbeePowerOutlet(ZIGBEE_OUTLET_ENDPOINT); diff --git a/libraries/Zigbee/examples/Zigbee_Power_Outlet/ci.json b/libraries/Zigbee/examples/Zigbee_Power_Outlet/ci.json index 23decd7127c..15d6190e4ae 100644 --- a/libraries/Zigbee/examples/Zigbee_Power_Outlet/ci.json +++ b/libraries/Zigbee/examples/Zigbee_Power_Outlet/ci.json @@ -1,7 +1,6 @@ { "fqbn_append": "PartitionScheme=zigbee_zczr,ZigbeeMode=zczr", "requires": [ - "CONFIG_SOC_IEEE802154_SUPPORTED=y", "CONFIG_ZB_ENABLED=y" ] } From 31d22e6ed05a26a7fd7765b9ae42fd292559690f Mon Sep 17 00:00:00 2001 From: SooD <45733639+syong0921@users.noreply.github.com> Date: Thu, 5 Jun 2025 00:49:09 +0900 Subject: [PATCH 036/173] fix: change geekble nano board setting (#11432) add PSRAM Setting --- boards.txt | 7 +++++++ 1 file changed, 7 insertions(+) diff --git a/boards.txt b/boards.txt index d92f057a78f..7e86af52a4f 100644 --- a/boards.txt +++ b/boards.txt @@ -41778,6 +41778,13 @@ Geekble_Nano_ESP32S3.menu.PartitionScheme.custom=Custom Geekble_Nano_ESP32S3.menu.PartitionScheme.custom.build.partitions= Geekble_Nano_ESP32S3.menu.PartitionScheme.custom.upload.maximum_size=16777216 +Geekble_Nano_ESP32S3.menu.PSRAM.disabled=Disabled +Geekble_Nano_ESP32S3.menu.PSRAM.disabled.build.defines= +Geekble_Nano_ESP32S3.menu.PSRAM.disabled.build.psram_type=qspi +Geekble_Nano_ESP32S3.menu.PSRAM.enabled=Enabled +Geekble_Nano_ESP32S3.menu.PSRAM.enabled.build.defines=-DBOARD_HAS_PSRAM +Geekble_Nano_ESP32S3.menu.PSRAM.enabled.build.psram_type=qspi + Geekble_Nano_ESP32S3.menu.DebugLevel.none=None Geekble_Nano_ESP32S3.menu.DebugLevel.none.build.code_debug=0 Geekble_Nano_ESP32S3.menu.DebugLevel.error=Error From 1bac8de384c6e27abde8901b4fb3ca712d11cf1b Mon Sep 17 00:00:00 2001 From: SooD <45733639+syong0921@users.noreply.github.com> Date: Tue, 10 Jun 2025 16:29:11 +0900 Subject: [PATCH 037/173] fix: Updated the tools options for Geekble Mini (#11437) fix: Updated the tools options for Geekble Mini --- boards.txt | 38 -------------------------------------- 1 file changed, 38 deletions(-) diff --git a/boards.txt b/boards.txt index 7e86af52a4f..49a4d9a737d 100644 --- a/boards.txt +++ b/boards.txt @@ -41589,11 +41589,6 @@ Geekble_ESP32C3.build.boot=qio Geekble_ESP32C3.build.partitions=default Geekble_ESP32C3.build.defines= -Geekble_ESP32C3.menu.CDCOnBoot.default=Enabled -Geekble_ESP32C3.menu.CDCOnBoot.default.build.cdc_on_boot=1 -Geekble_ESP32C3.menu.CDCOnBoot.cdc=Disabled -Geekble_ESP32C3.menu.CDCOnBoot.cdc.build.cdc_on_boot=0 - Geekble_ESP32C3.menu.PartitionScheme.default=Default 4MB with spiffs (1.2MB APP/1.5MB SPIFFS) Geekble_ESP32C3.menu.PartitionScheme.default.build.partitions=default Geekble_ESP32C3.menu.PartitionScheme.defaultffat=Default 4MB with ffat (1.2MB APP/1.5MB FATFS) @@ -41614,39 +41609,6 @@ Geekble_ESP32C3.menu.PartitionScheme.huge_app=Huge APP (3MB No OTA/1MB SPIFFS) Geekble_ESP32C3.menu.PartitionScheme.huge_app.build.partitions=huge_app Geekble_ESP32C3.menu.PartitionScheme.huge_app.upload.maximum_size=3145728 -Geekble_ESP32C3.menu.CPUFreq.160=160MHz (WiFi) (Default) -Geekble_ESP32C3.menu.CPUFreq.160.build.f_cpu=160000000L -Geekble_ESP32C3.menu.CPUFreq.80=80MHz (WiFi) -Geekble_ESP32C3.menu.CPUFreq.80.build.f_cpu=80000000L -Geekble_ESP32C3.menu.CPUFreq.40=40MHz -Geekble_ESP32C3.menu.CPUFreq.40.build.f_cpu=40000000L -Geekble_ESP32C3.menu.CPUFreq.20=20MHz -Geekble_ESP32C3.menu.CPUFreq.20.build.f_cpu=20000000L -Geekble_ESP32C3.menu.CPUFreq.10=10MHz -Geekble_ESP32C3.menu.CPUFreq.10.build.f_cpu=10000000L - -Geekble_ESP32C3.menu.FlashMode.qio=QIO (Default) -Geekble_ESP32C3.menu.FlashMode.qio.build.flash_mode=dio -Geekble_ESP32C3.menu.FlashMode.qio.build.boot=qio -Geekble_ESP32C3.menu.FlashMode.dio=DIO -Geekble_ESP32C3.menu.FlashMode.dio.build.flash_mode=dio -Geekble_ESP32C3.menu.FlashMode.dio.build.boot=dio -Geekble_ESP32C3.menu.FlashMode.qout=QOUT -Geekble_ESP32C3.menu.FlashMode.qout.build.flash_mode=dout -Geekble_ESP32C3.menu.FlashMode.qout.build.boot=qout -Geekble_ESP32C3.menu.FlashMode.dout=DOUT -Geekble_ESP32C3.menu.FlashMode.dout.build.flash_mode=dout - -Geekble_ESP32C3.menu.FlashFreq.80=80MHz (Default) -Geekble_ESP32C3.menu.FlashFreq.80.build.flash_freq=80m -Geekble_ESP32C3.menu.FlashFreq.40=40MHz -Geekble_ESP32C3.menu.FlashFreq.40.build.flash_freq=40m - -Geekble_ESP32C3.menu.FlashSize.4M=4MB (Default) -Geekble_ESP32C3.menu.FlashSize.4M.build.flash_size=4MB -Geekble_ESP32C3.menu.FlashSize.2M=2MB -Geekble_ESP32C3.menu.FlashSize.2M.build.flash_size=2MB - Geekble_ESP32C3.menu.UploadSpeed.921600=921600 (Default) Geekble_ESP32C3.menu.UploadSpeed.921600.upload.speed=921600 Geekble_ESP32C3.menu.UploadSpeed.115200=115200 From 610d951f9d1ddf690b786600bccddb515f1f0b5b Mon Sep 17 00:00:00 2001 From: Jason2866 <24528715+Jason2866@users.noreply.github.com> Date: Tue, 10 Jun 2025 09:29:29 +0200 Subject: [PATCH 038/173] include "esp_bt.h" only when existing (#11438) --- cores/esp32/esp32-hal-misc.c | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/cores/esp32/esp32-hal-misc.c b/cores/esp32/esp32-hal-misc.c index 594acd38153..aadd08ceffc 100644 --- a/cores/esp32/esp32-hal-misc.c +++ b/cores/esp32/esp32-hal-misc.c @@ -25,7 +25,7 @@ #include "esp_ota_ops.h" #endif //CONFIG_APP_ROLLBACK_ENABLE #include "esp_private/startup_internal.h" -#if defined(CONFIG_BT_BLUEDROID_ENABLED) && SOC_BT_SUPPORTED +#if defined(CONFIG_BT_BLUEDROID_ENABLED) && SOC_BT_SUPPORTED && __has_include("esp_bt.h") #include "esp_bt.h" #endif //CONFIG_BT_BLUEDROID_ENABLED #include From ee347baa7dca0cbe87dbd579465bbe00924f9b03 Mon Sep 17 00:00:00 2001 From: i3water <121024123@qq.com> Date: Tue, 10 Jun 2025 15:29:46 +0800 Subject: [PATCH 039/173] feat(boards): update wifiduinov2&wifiduino32s3 boards setting (#11440) * update wifiduinov2&wifiduino32s3 boards setting * fix wifiduinov2&wifiduino32s3 build board error. * fix wifiduinov2(esp32c3) board setting * fix wifiduinov2(esp32c3) cdc on boot default setting. * fix wifiduino32s3 spi pin set * change wifiduino32s3 spi pin to spi1 * remove 32Mb flash size * ci(pre-commit): Apply automatic fixes --------- Co-authored-by: pre-commit-ci-lite[bot] <117423508+pre-commit-ci-lite[bot]@users.noreply.github.com> --- boards.txt | 46 +++++++++---------- variants/wifiduino32s3/pins_arduino.h | 66 ++++++++++++++++----------- 2 files changed, 62 insertions(+), 50 deletions(-) diff --git a/boards.txt b/boards.txt index 49a4d9a737d..fefc6cdfb71 100644 --- a/boards.txt +++ b/boards.txt @@ -31513,16 +31513,16 @@ wifiduino32c3.build.target=esp wifiduino32c3.build.mcu=esp32c3 wifiduino32c3.build.core=esp32 wifiduino32c3.build.variant=wifiduinov2 -wifiduino32c3.build.board=WiFiduinoV2 +wifiduino32c3.build.board=WIFIDUINOV2 wifiduino32c3.build.bootloader_addr=0x0 wifiduino32c3.build.cdc_on_boot=0 wifiduino32c3.build.f_cpu=160000000L wifiduino32c3.build.flash_size=4MB wifiduino32c3.build.flash_freq=80m -wifiduino32c3.build.flash_mode=qio -wifiduino32c3.build.boot=qio -wifiduino32c3.build.partitions=default +wifiduino32c3.build.flash_mode=dio +wifiduino32c3.build.boot=dio +wifiduino32c3.build.partitions=no_ota wifiduino32c3.build.defines= wifiduino32c3.menu.CDCOnBoot.default=Disabled @@ -31530,6 +31530,9 @@ wifiduino32c3.menu.CDCOnBoot.default.build.cdc_on_boot=0 wifiduino32c3.menu.CDCOnBoot.cdc=Enabled wifiduino32c3.menu.CDCOnBoot.cdc.build.cdc_on_boot=1 +wifiduino32c3.menu.PartitionScheme.no_ota=No OTA (2MB APP/2MB SPIFFS) +wifiduino32c3.menu.PartitionScheme.no_ota.build.partitions=no_ota +wifiduino32c3.menu.PartitionScheme.no_ota.upload.maximum_size=2097152 wifiduino32c3.menu.PartitionScheme.default=Default 4MB with spiffs (1.2MB APP/1.5MB SPIFFS) wifiduino32c3.menu.PartitionScheme.default.build.partitions=default wifiduino32c3.menu.PartitionScheme.defaultffat=Default 4MB with ffat (1.2MB APP/1.5MB FATFS) @@ -31539,9 +31542,6 @@ wifiduino32c3.menu.PartitionScheme.default_8MB.build.partitions=default_8MB wifiduino32c3.menu.PartitionScheme.default_8MB.upload.maximum_size=3342336 wifiduino32c3.menu.PartitionScheme.minimal=Minimal (1.3MB APP/700KB SPIFFS) wifiduino32c3.menu.PartitionScheme.minimal.build.partitions=minimal -wifiduino32c3.menu.PartitionScheme.no_ota=No OTA (2MB APP/2MB SPIFFS) -wifiduino32c3.menu.PartitionScheme.no_ota.build.partitions=no_ota -wifiduino32c3.menu.PartitionScheme.no_ota.upload.maximum_size=2097152 wifiduino32c3.menu.PartitionScheme.noota_3g=No OTA (1MB APP/3MB SPIFFS) wifiduino32c3.menu.PartitionScheme.noota_3g.build.partitions=noota_3g wifiduino32c3.menu.PartitionScheme.noota_3g.upload.maximum_size=1048576 @@ -31584,12 +31584,12 @@ wifiduino32c3.menu.CPUFreq.20.build.f_cpu=20000000L wifiduino32c3.menu.CPUFreq.10=10MHz wifiduino32c3.menu.CPUFreq.10.build.f_cpu=10000000L -wifiduino32c3.menu.FlashMode.qio=QIO -wifiduino32c3.menu.FlashMode.qio.build.flash_mode=dio -wifiduino32c3.menu.FlashMode.qio.build.boot=qio wifiduino32c3.menu.FlashMode.dio=DIO wifiduino32c3.menu.FlashMode.dio.build.flash_mode=dio wifiduino32c3.menu.FlashMode.dio.build.boot=dio +wifiduino32c3.menu.FlashMode.qio=QIO +wifiduino32c3.menu.FlashMode.qio.build.flash_mode=dio +wifiduino32c3.menu.FlashMode.qio.build.boot=qio wifiduino32c3.menu.FlashFreq.80=80MHz wifiduino32c3.menu.FlashFreq.80.build.flash_freq=80m @@ -31665,34 +31665,34 @@ wifiduino32s3.build.target=esp32s3 wifiduino32s3.build.mcu=esp32s3 wifiduino32s3.build.core=esp32 wifiduino32s3.build.variant=wifiduino32s3 -wifiduino32s3.build.board=WiFiduino32S3 +wifiduino32s3.build.board=WIFIDUINO32S3 wifiduino32s3.build.usb_mode=1 wifiduino32s3.build.cdc_on_boot=0 wifiduino32s3.build.msc_on_boot=0 wifiduino32s3.build.dfu_on_boot=0 wifiduino32s3.build.f_cpu=240000000L -wifiduino32s3.build.flash_size=4MB +wifiduino32s3.build.flash_size=16MB wifiduino32s3.build.flash_freq=80m wifiduino32s3.build.flash_mode=dio wifiduino32s3.build.boot=qio wifiduino32s3.build.boot_freq=80m -wifiduino32s3.build.partitions=default +wifiduino32s3.build.partitions=app3M_fat9M_16MB wifiduino32s3.build.defines= wifiduino32s3.build.loop_core= wifiduino32s3.build.event_core= -wifiduino32s3.build.psram_type=qspi +wifiduino32s3.build.psram_type=opi wifiduino32s3.build.memory_type={build.boot}_{build.psram_type} +wifiduino32s3.menu.PSRAM.opi=OPI PSRAM +wifiduino32s3.menu.PSRAM.opi.build.defines=-DBOARD_HAS_PSRAM +wifiduino32s3.menu.PSRAM.opi.build.psram_type=opi wifiduino32s3.menu.PSRAM.disabled=Disabled wifiduino32s3.menu.PSRAM.disabled.build.defines= wifiduino32s3.menu.PSRAM.disabled.build.psram_type=qspi wifiduino32s3.menu.PSRAM.enabled=QSPI PSRAM wifiduino32s3.menu.PSRAM.enabled.build.defines=-DBOARD_HAS_PSRAM wifiduino32s3.menu.PSRAM.enabled.build.psram_type=qspi -wifiduino32s3.menu.PSRAM.opi=OPI PSRAM -wifiduino32s3.menu.PSRAM.opi.build.defines=-DBOARD_HAS_PSRAM -wifiduino32s3.menu.PSRAM.opi.build.psram_type=opi wifiduino32s3.menu.FlashMode.qio=QIO 80MHz wifiduino32s3.menu.FlashMode.qio.build.flash_mode=dio @@ -31715,12 +31715,10 @@ wifiduino32s3.menu.FlashMode.opi.build.boot=opi wifiduino32s3.menu.FlashMode.opi.build.boot_freq=80m wifiduino32s3.menu.FlashMode.opi.build.flash_freq=80m -wifiduino32s3.menu.FlashSize.4M=4MB (32Mb) -wifiduino32s3.menu.FlashSize.4M.build.flash_size=4MB -wifiduino32s3.menu.FlashSize.8M=8MB (64Mb) -wifiduino32s3.menu.FlashSize.8M.build.flash_size=8MB wifiduino32s3.menu.FlashSize.16M=16MB (128Mb) wifiduino32s3.menu.FlashSize.16M.build.flash_size=16MB +wifiduino32s3.menu.FlashSize.8M=8MB (64Mb) +wifiduino32s3.menu.FlashSize.8M.build.flash_size=8MB #wifiduino32s3.menu.FlashSize.32M=32MB (256Mb) #wifiduino32s3.menu.FlashSize.32M.build.flash_size=32MB @@ -31761,6 +31759,9 @@ wifiduino32s3.menu.UploadMode.cdc=USB-OTG CDC (TinyUSB) wifiduino32s3.menu.UploadMode.cdc.upload.use_1200bps_touch=true wifiduino32s3.menu.UploadMode.cdc.upload.wait_for_upload_port=true +wifiduino32s3.menu.PartitionScheme.app3M_fat9M_16MB=16M Flash (3MB APP/9.9MB FATFS) +wifiduino32s3.menu.PartitionScheme.app3M_fat9M_16MB.build.partitions=app3M_fat9M_16MB +wifiduino32s3.menu.PartitionScheme.app3M_fat9M_16MB.upload.maximum_size=3145728 wifiduino32s3.menu.PartitionScheme.default=Default 4MB with spiffs (1.2MB APP/1.5MB SPIFFS) wifiduino32s3.menu.PartitionScheme.default.build.partitions=default wifiduino32s3.menu.PartitionScheme.defaultffat=Default 4MB with ffat (1.2MB APP/1.5MB FATFS) @@ -31791,9 +31792,6 @@ wifiduino32s3.menu.PartitionScheme.min_spiffs.upload.maximum_size=1966080 wifiduino32s3.menu.PartitionScheme.fatflash=16M Flash (2MB APP/12.5MB FATFS) wifiduino32s3.menu.PartitionScheme.fatflash.build.partitions=ffat wifiduino32s3.menu.PartitionScheme.fatflash.upload.maximum_size=2097152 -wifiduino32s3.menu.PartitionScheme.app3M_fat9M_16MB=16M Flash (3MB APP/9.9MB FATFS) -wifiduino32s3.menu.PartitionScheme.app3M_fat9M_16MB.build.partitions=app3M_fat9M_16MB -wifiduino32s3.menu.PartitionScheme.app3M_fat9M_16MB.upload.maximum_size=3145728 wifiduino32s3.menu.PartitionScheme.rainmaker=RainMaker 4MB wifiduino32s3.menu.PartitionScheme.rainmaker.build.partitions=rainmaker wifiduino32s3.menu.PartitionScheme.rainmaker.upload.maximum_size=1966080 diff --git a/variants/wifiduino32s3/pins_arduino.h b/variants/wifiduino32s3/pins_arduino.h index d26e415910e..4fc08139c0e 100644 --- a/variants/wifiduino32s3/pins_arduino.h +++ b/variants/wifiduino32s3/pins_arduino.h @@ -2,43 +2,57 @@ #define Pins_Arduino_h #include - -#define USB_VID 0x303a -#define USB_PID 0x1001 - -// No USER LED or NeoLED - -static const uint8_t TX = 45; +#include "soc/soc_caps.h" + +#define USB_VID 0x303a +#define USB_PID 0x1001 +#define USB_MANUFACTURER "openjumper" +#define USB_PRODUCT "Wifiduino32-S3" +#define USB_SERIAL "" // Empty string for MAC address + +// Some boards have too low voltage on this pin (board design bug) +// Use different pin with 3V and connect with 48 +// and change this setup for the chosen pin (for example 38) +#define PIN_RGB_LED 48 +// BUILTIN_LED can be used in new Arduino API digitalWrite() like in Blink.ino +static const uint8_t LED_BUILTIN = SOC_GPIO_PIN_COUNT + PIN_RGB_LED; +#define BUILTIN_LED LED_BUILTIN // backward compatibility +#define LED_BUILTIN LED_BUILTIN // allow testing #ifdef LED_BUILTIN +// RGB_BUILTIN and RGB_BRIGHTNESS can be used in new Arduino API rgbLedWrite() +#define RGB_BUILTIN LED_BUILTIN +#define RGB_BRIGHTNESS 64 + +static const uint8_t TX = 43; static const uint8_t RX = 44; static const uint8_t SDA = 4; static const uint8_t SCL = 5; -static const uint8_t SS = 46; -static const uint8_t MOSI = 3; -static const uint8_t MISO = 20; -static const uint8_t SCK = 19; +static const uint8_t SS = 10; +static const uint8_t MOSI = 11; +static const uint8_t MISO = 13; +static const uint8_t SCK = 12; -static const uint8_t A0 = 7; -static const uint8_t A1 = 6; +static const uint8_t A0 = 0; +static const uint8_t A1 = 1; static const uint8_t A2 = 2; -static const uint8_t A3 = 1; +static const uint8_t A3 = 3; static const uint8_t A4 = 4; static const uint8_t A5 = 5; static const uint8_t D0 = 44; -static const uint8_t D1 = 45; -static const uint8_t D2 = 42; -static const uint8_t D3 = 41; -static const uint8_t D4 = 0; -static const uint8_t D5 = 45; -static const uint8_t D6 = 48; -static const uint8_t D7 = 47; +static const uint8_t D1 = 43; +static const uint8_t D2 = 45; +static const uint8_t D3 = 46; +static const uint8_t D4 = 47; +static const uint8_t D5 = 48; +static const uint8_t D6 = 18; +static const uint8_t D7 = 17; static const uint8_t D8 = 21; -static const uint8_t D9 = 14; -static const uint8_t D10 = 46; -static const uint8_t D11 = 3; -static const uint8_t D12 = 20; -static const uint8_t D13 = 19; +static const uint8_t D9 = 42; +static const uint8_t D10 = 41; +static const uint8_t D11 = 40; +static const uint8_t D12 = 38; +static const uint8_t D13 = 39; #endif /* Pins_Arduino_h */ From af47bd30089bee71849b634b97dd329e6564a10a Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jan=20Proch=C3=A1zka?= <90197375+P-R-O-C-H-Y@users.noreply.github.com> Date: Tue, 10 Jun 2025 09:59:56 +0200 Subject: [PATCH 040/173] fix(ci): Process only needed files in publish sizes (#11439) --- .github/workflows/publishsizes.yml | 19 ++++++++++--------- 1 file changed, 10 insertions(+), 9 deletions(-) diff --git a/.github/workflows/publishsizes.yml b/.github/workflows/publishsizes.yml index 69c18cf1835..8ff591e052b 100644 --- a/.github/workflows/publishsizes.yml +++ b/.github/workflows/publishsizes.yml @@ -44,16 +44,17 @@ jobs: gh api "$artifacts_url" -q '.artifacts[] | [.name, .archive_download_url] | @tsv' | while read artifact do IFS=$'\t' read name url <<< "$artifact" - gh api $url > "$name.zip" - unzip -j "$name.zip" -d "temp_$name" - if [[ "$name" == "pr_number" ]]; then - mv "temp_$name"/* sizes-report - elif [[ "$name" == "pr_cli"* ]]; then - mv "temp_$name"/* sizes-report/pr - else - mv "temp_$name"/* sizes-report + # Only process pr_number and pr_cli_compile artifacts + if [[ "$name" == "pr_number" || "$name" =~ ^pr_cli_compile_[0-9]+$ ]]; then + gh api $url > "$name.zip" + unzip -o -j "$name.zip" -d "temp_$name" + if [[ "$name" == "pr_number" ]]; then + mv "temp_$name"/* sizes-report + elif [[ "$name" =~ ^pr_cli_compile_[0-9]+$ ]]; then + mv "temp_$name"/* sizes-report/pr + fi + rm -r "temp_$name" fi - rm -r "temp_$name" done echo "Contents of parent directory:" ls -R .. From 0007815a1148c7a3d24a3f6986739bc2b26eb5c7 Mon Sep 17 00:00:00 2001 From: Me No Dev Date: Tue, 10 Jun 2025 11:00:20 +0300 Subject: [PATCH 041/173] feat(p4): Add 32MB Flash Partitions to ESP32-P4 (#11453) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit * feat(p4): Add 32MB Flash Partitions to ESP32-P4 * feat(p4): Add 32MB flash size option --------- Co-authored-by: Jan Procházka <90197375+P-R-O-C-H-Y@users.noreply.github.com> --- boards.txt | 11 +++++++++++ 1 file changed, 11 insertions(+) diff --git a/boards.txt b/boards.txt index fefc6cdfb71..15a75eaad8b 100644 --- a/boards.txt +++ b/boards.txt @@ -282,6 +282,15 @@ esp32p4.menu.PartitionScheme.fatflash.upload.maximum_size=2097152 esp32p4.menu.PartitionScheme.app3M_fat9M_16MB=16M Flash (3MB APP/9.9MB FATFS) esp32p4.menu.PartitionScheme.app3M_fat9M_16MB.build.partitions=app3M_fat9M_16MB esp32p4.menu.PartitionScheme.app3M_fat9M_16MB.upload.maximum_size=3145728 +esp32p4.menu.PartitionScheme.app5M_fat24M_32MB=32M Flash (4.8MB APP/22MB FATFS) +esp32p4.menu.PartitionScheme.app5M_fat24M_32MB.build.partitions=large_fat_32MB +esp32p4.menu.PartitionScheme.app5M_fat24M_32MB.upload.maximum_size=4718592 +esp32p4.menu.PartitionScheme.app5M_little24M_32MB=32M Flash (4.8MB APP/22MB LittleFS) +esp32p4.menu.PartitionScheme.app5M_little24M_32MB.build.partitions=large_littlefs_32MB +esp32p4.menu.PartitionScheme.app5M_little24M_32MB.upload.maximum_size=4718592 +esp32p4.menu.PartitionScheme.app13M_data7M_32MB=32M Flash (13MB APP/6.75MB SPIFFS) +esp32p4.menu.PartitionScheme.app13M_data7M_32MB.build.partitions=default_32MB +esp32p4.menu.PartitionScheme.app13M_data7M_32MB.upload.maximum_size=13107200 esp32p4.menu.PartitionScheme.custom=Custom esp32p4.menu.PartitionScheme.custom.build.partitions= esp32p4.menu.PartitionScheme.custom.upload.maximum_size=16777216 @@ -314,6 +323,8 @@ esp32p4.menu.FlashSize.2M.build.flash_size=2MB esp32p4.menu.FlashSize.2M.build.partitions=minimal esp32p4.menu.FlashSize.16M=16MB (128Mb) esp32p4.menu.FlashSize.16M.build.flash_size=16MB +esp32p4.menu.FlashSize.32M=32MB (256Mb) +esp32p4.menu.FlashSize.32M.build.flash_size=32MB esp32p4.menu.UploadSpeed.921600=921600 esp32p4.menu.UploadSpeed.921600.upload.speed=921600 From 89ff4653286206be29fe49e3c333f14065801616 Mon Sep 17 00:00:00 2001 From: Lucas Saavedra Vaz <32426024+lucasssvaz@users.noreply.github.com> Date: Tue, 10 Jun 2025 05:19:32 -0300 Subject: [PATCH 042/173] feat(esptool): Upgrade to esptool v5 (#11433) * feat(esptool): Upgrade to esptool v5 * fix(script): Update script for better handling of esptool * fix(script): Get proper download url * fix(script): Apply copilot suggestions --- .github/scripts/update_esptool.py | 236 ++++++++++++++++++++++ package/package_esp32_index.template.json | 70 +++---- platform.txt | 12 +- 3 files changed, 277 insertions(+), 41 deletions(-) create mode 100644 .github/scripts/update_esptool.py diff --git a/.github/scripts/update_esptool.py b/.github/scripts/update_esptool.py new file mode 100644 index 00000000000..dd5de5526c3 --- /dev/null +++ b/.github/scripts/update_esptool.py @@ -0,0 +1,236 @@ +#!/usr/bin/env python3 + +# This script is used to re-package the esptool if needed and update the JSON file +# for the Arduino ESP32 platform. +# +# The script has only been tested on macOS. +# +# For regular esptool releases, the generated packages already contain the correct permissions, +# extensions and are uploaded to the GitHub release assets. In this case, the script will only +# update the JSON file with the information from the GitHub release. +# +# The script can be used in two modes: +# 1. Local build: The build artifacts must be already downloaded and extracted in the base_folder. +# This is useful for esptool versions that are not yet released and that are grabbed from the +# GitHub build artifacts. +# 2. Release build: The script will get the release information from GitHub and update the JSON file. +# This is useful for esptool versions that are already released and that are uploaded to the +# GitHub release assets. +# +# For local build, the artifacts must be already downloaded and extracted in the base_folder +# set with the -l option. +# For example, a base folder "esptool" should contain the following folders extracted directly +# from the GitHub build artifacts: +# esptool/esptool-linux-aarch64 +# esptool/esptool-linux-amd64 +# esptool/esptool-linux-armv7 +# esptool/esptool-macos-amd64 +# esptool/esptool-macos-arm64 +# esptool/esptool-windows-amd64 + +import argparse +import json +import os +import shutil +import stat +import tarfile +import zipfile +import hashlib +import requests +from pathlib import Path + +def compute_sha256(filepath): + sha256 = hashlib.sha256() + with open(filepath, "rb") as f: + for block in iter(lambda: f.read(4096), b""): + sha256.update(block) + return f"SHA-256:{sha256.hexdigest()}" + +def get_file_size(filepath): + return os.path.getsize(filepath) + +def update_json_for_host(tmp_json_path, version, host, url, archiveFileName, checksum, size): + with open(tmp_json_path) as f: + data = json.load(f) + + for pkg in data.get("packages", []): + for tool in pkg.get("tools", []): + if tool.get("name") == "esptool_py": + tool["version"] = version + + if url is None: + # If the URL is not set, we need to find the old URL and update it + for system in tool.get("systems", []): + if system.get("host") == host: + url = system.get("url").replace(system.get("archiveFileName"), archiveFileName) + break + else: + print(f"No old URL found for host {host}. Using empty URL.") + url = "" + + # Preserve existing systems order and update or append the new system + systems = tool.get("systems", []) + system_updated = False + for i, system in enumerate(systems): + if system.get("host") == host: + systems[i] = { + "host": host, + "url": url, + "archiveFileName": archiveFileName, + "checksum": checksum, + "size": str(size), + } + system_updated = True + break + + if not system_updated: + systems.append({ + "host": host, + "url": url, + "archiveFileName": archiveFileName, + "checksum": checksum, + "size": str(size), + }) + tool["systems"] = systems + + with open(tmp_json_path, "w") as f: + json.dump(data, f, indent=2, sort_keys=False, ensure_ascii=False) + f.write("\n") + +def update_tools_dependencies(tmp_json_path, version): + with open(tmp_json_path) as f: + data = json.load(f) + + for pkg in data.get("packages", []): + for platform in pkg.get("platforms", []): + for dep in platform.get("toolsDependencies", []): + if dep.get("name") == "esptool_py": + dep["version"] = version + + with open(tmp_json_path, "w") as f: + json.dump(data, f, indent=2, sort_keys=False, ensure_ascii=False) + f.write("\n") + +def create_archives(version, base_folder): + archive_files = [] + + for dirpath in Path(base_folder).glob("esptool-*"): + if not dirpath.is_dir(): + continue + + base = dirpath.name[len("esptool-"):] + + if "windows" in dirpath.name: + zipfile_name = f"esptool-v{version}-{base}.zip" + print(f"Creating {zipfile_name} from {dirpath} ...") + with zipfile.ZipFile(zipfile_name, "w", zipfile.ZIP_DEFLATED) as zipf: + for root, _, files in os.walk(dirpath): + for file in files: + full_path = os.path.join(root, file) + zipf.write(full_path, os.path.relpath(full_path, start=dirpath)) + archive_files.append(zipfile_name) + else: + tarfile_name = f"esptool-v{version}-{base}.tar.gz" + print(f"Creating {tarfile_name} from {dirpath} ...") + for root, dirs, files in os.walk(dirpath): + for name in dirs + files: + os.chmod(os.path.join(root, name), stat.S_IRUSR | stat.S_IWUSR | stat.S_IXUSR | + stat.S_IRGRP | stat.S_IXGRP | + stat.S_IROTH | stat.S_IXOTH) + with tarfile.open(tarfile_name, "w:gz") as tar: + tar.add(dirpath, arcname=dirpath.name) + archive_files.append(tarfile_name) + + return archive_files + +def determine_hosts(archive_name): + if "linux-amd64" in archive_name: + return ["x86_64-pc-linux-gnu"] + elif "linux-armv7" in archive_name: + return ["arm-linux-gnueabihf"] + elif "linux-aarch64" in archive_name: + return ["aarch64-linux-gnu"] + elif "macos-amd64" in archive_name: + return ["x86_64-apple-darwin"] + elif "macos-arm64" in archive_name: + return ["arm64-apple-darwin"] + elif "windows-amd64" in archive_name: + return ["x86_64-mingw32", "i686-mingw32"] + else: + return [] + +def update_json_from_local_build(tmp_json_path, version, base_folder, archive_files): + for archive in archive_files: + print(f"Processing archive: {archive}") + hosts = determine_hosts(archive) + if not hosts: + print(f"Skipping unknown archive type: {archive}") + continue + + archive_path = Path(archive) + checksum = compute_sha256(archive_path) + size = get_file_size(archive_path) + + for host in hosts: + update_json_for_host(tmp_json_path, version, host, None, archive_path.name, checksum, size) + +def update_json_from_release(tmp_json_path, version, release_info): + assets = release_info.get("assets", []) + for asset in assets: + if (asset.get("name").endswith(".tar.gz") or asset.get("name").endswith(".zip")) and "esptool" in asset.get("name"): + asset_fname = asset.get("name") + print(f"Processing asset: {asset_fname}") + hosts = determine_hosts(asset_fname) + if not hosts: + print(f"Skipping unknown archive type: {asset_fname}") + continue + + asset_url = asset.get("browser_download_url") + asset_checksum = asset.get("digest") + asset_size = asset.get("size") + if asset_checksum is None: + asset_checksum = "" + print(f"Asset {asset_fname} has no checksum. Please set the checksum in the JSON file.") + + for host in hosts: + update_json_for_host(tmp_json_path, version, host, asset_url, asset_fname, asset_checksum, asset_size) + +def get_release_info(version): + url = f"https://api.github.com/repos/espressif/esptool/releases/tags/v{version}" + response = requests.get(url) + response.raise_for_status() + return response.json() + +def main(): + parser = argparse.ArgumentParser(description="Repack esptool and update JSON metadata.") + parser.add_argument("version", help="Version of the esptool (e.g. 5.0.dev1)") + parser.add_argument("-l", "--local", dest="base_folder", help="Enable local build mode and set the base folder with unpacked artifacts") + args = parser.parse_args() + + script_dir = Path(__file__).resolve().parent + json_path = (script_dir / "../../package/package_esp32_index.template.json").resolve() + tmp_json_path = Path(str(json_path) + ".tmp") + shutil.copy(json_path, tmp_json_path) + + local_build = args.base_folder is not None + + if local_build: + os.chdir(args.base_folder) + os.environ['COPYFILE_DISABLE'] = 'true' # this disables including resource forks in tar files on macOS + # Clear any existing archive files + for file in Path(args.base_folder).glob("esptool-*.*"): + file.unlink() + archive_files = create_archives(args.version, args.base_folder) + update_json_from_local_build(tmp_json_path, args.version, args.base_folder, archive_files) + else: + release_info = get_release_info(args.version) + update_json_from_release(tmp_json_path, args.version, release_info) + + print(f"Updating esptool version fields to {args.version}") + update_tools_dependencies(tmp_json_path, args.version) + + shutil.move(tmp_json_path, json_path) + print(f"Done. JSON updated at {json_path}") + +if __name__ == "__main__": + main() diff --git a/package/package_esp32_index.template.json b/package/package_esp32_index.template.json index eecc7c10788..3ae2ff09ee6 100644 --- a/package/package_esp32_index.template.json +++ b/package/package_esp32_index.template.json @@ -81,7 +81,7 @@ { "packager": "esp32", "name": "esptool_py", - "version": "4.9.dev3" + "version": "5.0.dev1" }, { "packager": "esp32", @@ -469,56 +469,56 @@ }, { "name": "esptool_py", - "version": "4.9.dev3", + "version": "5.0.dev1", "systems": [ { - "host": "x86_64-pc-linux-gnu", - "url": "https://github.com/espressif/arduino-esp32/releases/download/3.1.0-RC3/esptool-v4.9.dev3-linux-amd64.tar.gz", - "archiveFileName": "esptool-v4.9.dev3-linux-amd64.tar.gz", - "checksum": "SHA-256:4ecaf51836cbf4ea3c19840018bfef3b0b8cd8fc3c95f6e1e043ca5bbeab9bf0", - "size": "64958202" + "host": "aarch64-linux-gnu", + "url": "https://github.com/espressif/arduino-esp32/releases/download/3.2.0/esptool-v5.0.dev1-linux-aarch64.tar.gz", + "archiveFileName": "esptool-v5.0.dev1-linux-aarch64.tar.gz", + "checksum": "SHA-256:bfafa7a7723ebbabfd8b6e3ca5ae00bfead0331de923754aeddb43b2c116a078", + "size": "58241736" }, { - "host": "arm-linux-gnueabihf", - "url": "https://github.com/espressif/arduino-esp32/releases/download/3.1.0-RC3/esptool-v4.9.dev3-linux-armv7.tar.gz", - "archiveFileName": "esptool-v4.9.dev3-linux-armv7.tar.gz", - "checksum": "SHA-256:fff818573bce483ee793ac83c8211f6abf764aa3350f198228859f696a0a0b36", - "size": "31530030" + "host": "x86_64-pc-linux-gnu", + "url": "https://github.com/espressif/arduino-esp32/releases/download/3.2.0/esptool-v5.0.dev1-linux-amd64.tar.gz", + "archiveFileName": "esptool-v5.0.dev1-linux-amd64.tar.gz", + "checksum": "SHA-256:acd0486e96586b99d053a1479acbbbfcae8667227c831cdc53a171f9ccfa27ee", + "size": "100740042" }, { - "host": "aarch64-linux-gnu", - "url": "https://github.com/espressif/arduino-esp32/releases/download/3.1.0-RC3/esptool-v4.9.dev3-linux-aarch64.tar.gz", - "archiveFileName": "esptool-v4.9.dev3-linux-aarch64.tar.gz", - "checksum": "SHA-256:5b274bdff2f62e6a07c3c1dfa51b1128924621f661747eca3dbe0f77972f2f06", - "size": "33663882" + "host": "arm-linux-gnueabihf", + "url": "https://github.com/espressif/arduino-esp32/releases/download/3.2.0/esptool-v5.0.dev1-linux-armv7.tar.gz", + "archiveFileName": "esptool-v5.0.dev1-linux-armv7.tar.gz", + "checksum": "SHA-256:ea77a38681506761bbb7b0b39c130811ed565667b67ebbdb4d6dcc6cb6e07368", + "size": "53451939" }, { "host": "x86_64-apple-darwin", - "url": "https://github.com/espressif/arduino-esp32/releases/download/3.1.0-RC3/esptool-v4.9.dev3-macos-amd64.tar.gz", - "archiveFileName": "esptool-v4.9.dev3-macos-amd64.tar.gz", - "checksum": "SHA-256:c733c83b58fcf5f642fbb2fddb8ff24640c2c785126cba0821fb70c4a5ceea7a", - "size": "32767836" + "url": "https://github.com/espressif/arduino-esp32/releases/download/3.2.0/esptool-v5.0.dev1-macos-amd64.tar.gz", + "archiveFileName": "esptool-v5.0.dev1-macos-amd64.tar.gz", + "checksum": "SHA-256:900a8e90731208bee96647e0e207a43612b9452c2120c4fdc0ff4c6be226257b", + "size": "59631998" }, { "host": "arm64-apple-darwin", - "url": "https://github.com/espressif/arduino-esp32/releases/download/3.1.0-RC3/esptool-v4.9.dev3-macos-arm64.tar.gz", - "archiveFileName": "esptool-v4.9.dev3-macos-arm64.tar.gz", - "checksum": "SHA-256:83c195a15981e6a5e7a130db2ccfb21e2d8093912e5b003681f9a5abadd71af7", - "size": "30121441" + "url": "https://github.com/espressif/arduino-esp32/releases/download/3.2.0/esptool-v5.0.dev1-macos-arm64.tar.gz", + "archiveFileName": "esptool-v5.0.dev1-macos-arm64.tar.gz", + "checksum": "SHA-256:3653f4de73cb4fc6a25351eaf663708e91c65ae3265d75bd54ca4315a4350bb4", + "size": "56349992" }, { - "host": "i686-mingw32", - "url": "https://github.com/espressif/arduino-esp32/releases/download/3.1.0-RC3/esptool-v4.9.dev3-win64.zip", - "archiveFileName": "esptool-v4.9.dev3-win64.zip", - "checksum": "SHA-256:890051a4fdc684ff6f4af18d0bb27d274ca940ee0eef716a9455f8c64b25b215", - "size": "36072564" + "host": "x86_64-mingw32", + "url": "https://github.com/espressif/arduino-esp32/releases/download/3.2.0/esptool-v5.0.dev1-win64.zip", + "archiveFileName": "esptool-v5.0.dev1-win64.zip", + "checksum": "SHA-256:1e8fd89645daf94f2d4406ec73c9004e617ea921079515f9fd749205eece4d6d", + "size": "59102658" }, { - "host": "x86_64-mingw32", - "url": "https://github.com/espressif/arduino-esp32/releases/download/3.1.0-RC3/esptool-v4.9.dev3-win64.zip", - "archiveFileName": "esptool-v4.9.dev3-win64.zip", - "checksum": "SHA-256:890051a4fdc684ff6f4af18d0bb27d274ca940ee0eef716a9455f8c64b25b215", - "size": "36072564" + "host": "i686-mingw32", + "url": "https://github.com/espressif/arduino-esp32/releases/download/3.2.0/esptool-v5.0.dev1-win64.zip", + "archiveFileName": "esptool-v5.0.dev1-win64.zip", + "checksum": "SHA-256:1e8fd89645daf94f2d4406ec73c9004e617ea921079515f9fd749205eece4d6d", + "size": "59102658" } ] }, diff --git a/platform.txt b/platform.txt index f41a8b4a764..8bfa69a79e0 100644 --- a/platform.txt +++ b/platform.txt @@ -120,7 +120,7 @@ recipe.hooks.prebuild.2.pattern.windows=cmd /c if not exist "{build.path}\partit recipe.hooks.prebuild.3.pattern.windows=cmd /c if not exist "{build.path}\partitions.csv" COPY "{runtime.platform.path}\tools\partitions\{build.partitions}.csv" "{build.path}\partitions.csv" # Check if custom bootloader exist: source > variant > build.boot -recipe.hooks.prebuild.4.pattern_args=--chip {build.mcu} elf2image --flash_mode {build.flash_mode} --flash_freq {build.img_freq} --flash_size {build.flash_size} -o +recipe.hooks.prebuild.4.pattern_args=--chip {build.mcu} elf2image --flash-mode {build.flash_mode} --flash-freq {build.img_freq} --flash-size {build.flash_size} -o recipe.hooks.prebuild.4.pattern=/usr/bin/env bash -c "[ -f "{build.source.path}"/bootloader.bin ] && cp -f "{build.source.path}"/bootloader.bin "{build.path}"/{build.project_name}.bootloader.bin || ( [ -f "{build.variant.path}"/{build.custom_bootloader}.bin ] && cp "{build.variant.path}"/{build.custom_bootloader}.bin "{build.path}"/{build.project_name}.bootloader.bin || "{tools.esptool_py.path}"/{tools.esptool_py.cmd} {recipe.hooks.prebuild.4.pattern_args} "{build.path}"/{build.project_name}.bootloader.bin "{compiler.sdk.path}"/bin/bootloader_{build.boot}_{build.boot_freq}.elf )" recipe.hooks.prebuild.4.pattern.windows=cmd /c IF EXIST "{build.source.path}\bootloader.bin" ( COPY /y "{build.source.path}\bootloader.bin" "{build.path}\{build.project_name}.bootloader.bin" ) ELSE ( IF EXIST "{build.variant.path}\{build.custom_bootloader}.bin" ( COPY "{build.variant.path}\{build.custom_bootloader}.bin" "{build.path}\{build.project_name}.bootloader.bin" ) ELSE ( "{tools.esptool_py.path}\{tools.esptool_py.cmd}" {recipe.hooks.prebuild.4.pattern_args} "{build.path}\{build.project_name}.bootloader.bin" "{compiler.sdk.path}\bin\bootloader_{build.boot}_{build.boot_freq}.elf" ) ) @@ -164,7 +164,7 @@ recipe.c.combine.pattern="{compiler.path}{compiler.c.elf.cmd}" {compiler.c.elf.f recipe.objcopy.partitions.bin.pattern={tools.gen_esp32part.cmd} -q "{build.path}/partitions.csv" "{build.path}/{build.project_name}.partitions.bin" ## Create bin -recipe.objcopy.bin.pattern_args=--chip {build.mcu} elf2image --flash_mode "{build.flash_mode}" --flash_freq "{build.img_freq}" --flash_size "{build.flash_size}" --elf-sha256-offset 0xb0 -o "{build.path}/{build.project_name}.bin" "{build.path}/{build.project_name}.elf" +recipe.objcopy.bin.pattern_args=--chip {build.mcu} elf2image --flash-mode "{build.flash_mode}" --flash-freq "{build.img_freq}" --flash-size "{build.flash_size}" --elf-sha256-offset 0xb0 -o "{build.path}/{build.project_name}.bin" "{build.path}/{build.project_name}.elf" recipe.objcopy.bin.pattern="{tools.esptool_py.path}/{tools.esptool_py.cmd}" {recipe.objcopy.bin.pattern_args} ## Create Insights Firmware Package @@ -177,7 +177,7 @@ recipe.hooks.objcopy.postobjcopy.2.pattern=/usr/bin/env bash -c "[ ! -d "{build. recipe.hooks.objcopy.postobjcopy.2.pattern.windows=cmd /c if exist "{build.path}\libraries\ESP_SR" if exist "{compiler.sdk.path}\esp_sr\srmodels.bin" COPY /y "{compiler.sdk.path}\esp_sr\srmodels.bin" "{build.path}\srmodels.bin" # Create merged binary -recipe.hooks.objcopy.postobjcopy.3.pattern_args=--chip {build.mcu} merge_bin -o "{build.path}/{build.project_name}.merged.bin" --fill-flash-size {build.flash_size} --flash_mode keep --flash_freq keep --flash_size keep {build.bootloader_addr} "{build.path}/{build.project_name}.bootloader.bin" 0x8000 "{build.path}/{build.project_name}.partitions.bin" 0xe000 "{runtime.platform.path}/tools/partitions/boot_app0.bin" 0x10000 "{build.path}/{build.project_name}.bin" +recipe.hooks.objcopy.postobjcopy.3.pattern_args=--chip {build.mcu} merge-bin -o "{build.path}/{build.project_name}.merged.bin" --pad-to-size {build.flash_size} --flash-mode keep --flash-freq keep --flash-size keep {build.bootloader_addr} "{build.path}/{build.project_name}.bootloader.bin" 0x8000 "{build.path}/{build.project_name}.partitions.bin" 0xe000 "{runtime.platform.path}/tools/partitions/boot_app0.bin" 0x10000 "{build.path}/{build.project_name}.bin" recipe.hooks.objcopy.postobjcopy.3.pattern="{tools.esptool_py.path}/{tools.esptool_py.cmd}" {recipe.hooks.objcopy.postobjcopy.3.pattern_args} ## Save bin @@ -294,14 +294,14 @@ debug.additional_config=debug_config.{build.mcu} tools.esptool_py.upload.protocol=serial tools.esptool_py.upload.params.verbose= tools.esptool_py.upload.params.quiet= -tools.esptool_py.upload.pattern_args=--chip {build.mcu} --port "{serial.port}" --baud {upload.speed} {upload.flags} --before default_reset --after hard_reset write_flash {upload.erase_cmd} -z --flash_mode keep --flash_freq keep --flash_size keep {build.bootloader_addr} "{build.path}/{build.project_name}.bootloader.bin" 0x8000 "{build.path}/{build.project_name}.partitions.bin" 0xe000 "{runtime.platform.path}/tools/partitions/boot_app0.bin" 0x10000 "{build.path}/{build.project_name}.bin" {upload.extra_flags} +tools.esptool_py.upload.pattern_args=--chip {build.mcu} --port "{serial.port}" --baud {upload.speed} {upload.flags} --before default-reset --after hard-reset write-flash {upload.erase_cmd} -z --flash-mode keep --flash-freq keep --flash-size keep {build.bootloader_addr} "{build.path}/{build.project_name}.bootloader.bin" 0x8000 "{build.path}/{build.project_name}.partitions.bin" 0xe000 "{runtime.platform.path}/tools/partitions/boot_app0.bin" 0x10000 "{build.path}/{build.project_name}.bin" {upload.extra_flags} tools.esptool_py.upload.pattern="{path}/{cmd}" {upload.pattern_args} ## Program Application ## ------------------- tools.esptool_py.program.params.verbose= tools.esptool_py.program.params.quiet= -tools.esptool_py.program.pattern_args=--chip {build.mcu} --port "{serial.port}" --baud {upload.speed} {upload.flags} --before default_reset --after hard_reset write_flash -z --flash_mode keep --flash_freq keep --flash_size keep 0x10000 "{build.path}/{build.project_name}.bin" +tools.esptool_py.program.pattern_args=--chip {build.mcu} --port "{serial.port}" --baud {upload.speed} {upload.flags} --before default-reset --after hard-reset write-flash -z --flash-mode keep --flash-freq keep --flash-size keep 0x10000 "{build.path}/{build.project_name}.bin" tools.esptool_py.program.pattern="{path}/{cmd}" {program.pattern_args} ## Erase Chip (before burning the bootloader) @@ -309,7 +309,7 @@ tools.esptool_py.program.pattern="{path}/{cmd}" {program.pattern_args} tools.esptool_py.erase.protocol=serial tools.esptool_py.erase.params.verbose= tools.esptool_py.erase.params.quiet= -tools.esptool_py.erase.pattern_args=--chip {build.mcu} --port "{serial.port}" --baud {upload.speed} {upload.flags} --before default_reset --after hard_reset erase_flash +tools.esptool_py.erase.pattern_args=--chip {build.mcu} --port "{serial.port}" --baud {upload.speed} {upload.flags} --before default-reset --after hard-reset erase-flash tools.esptool_py.erase.pattern="{path}/{cmd}" {erase.pattern_args} ## Burn Bootloader From 0aada091e1afb56bb4e2103297233b17a9463bde Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jan=20Proch=C3=A1zka?= <90197375+P-R-O-C-H-Y@users.noreply.github.com> Date: Tue, 10 Jun 2025 10:31:17 +0200 Subject: [PATCH 043/173] feat(zigbee): Support min/max setting for Analog EP (#11451) * feat(zigbee): Support min max for Analog EP * feat(zigbee): Use cfloat FLT_MAX * ci(pre-commit): Apply automatic fixes --------- Co-authored-by: pre-commit-ci-lite[bot] <117423508+pre-commit-ci-lite[bot]@users.noreply.github.com> --- .../Zigbee_Analog_Input_Output.ino | 3 + libraries/Zigbee/src/ep/ZigbeeAnalog.cpp | 84 +++++++++++++++++++ libraries/Zigbee/src/ep/ZigbeeAnalog.h | 4 + 3 files changed, 91 insertions(+) diff --git a/libraries/Zigbee/examples/Zigbee_Analog_Input_Output/Zigbee_Analog_Input_Output.ino b/libraries/Zigbee/examples/Zigbee_Analog_Input_Output/Zigbee_Analog_Input_Output.ino index 59c4b514db1..f1cc54bda64 100644 --- a/libraries/Zigbee/examples/Zigbee_Analog_Input_Output/Zigbee_Analog_Input_Output.ino +++ b/libraries/Zigbee/examples/Zigbee_Analog_Input_Output/Zigbee_Analog_Input_Output.ino @@ -72,6 +72,9 @@ void setup() { zbAnalogDevice.setAnalogOutputDescription("Fan Speed (RPM)"); zbAnalogDevice.setAnalogOutputResolution(1); + // Set the min and max values for the analog output which is used by HA to limit the range of the analog output + zbAnalogDevice.setAnalogOutputMinMax(-10000, 10000); //-10000 to 10000 RPM + // If analog output cluster is added, set callback function for analog output change zbAnalogDevice.onAnalogOutputChange(onAnalogOutputChange); diff --git a/libraries/Zigbee/src/ep/ZigbeeAnalog.cpp b/libraries/Zigbee/src/ep/ZigbeeAnalog.cpp index 893a9854ecc..309739d54c9 100644 --- a/libraries/Zigbee/src/ep/ZigbeeAnalog.cpp +++ b/libraries/Zigbee/src/ep/ZigbeeAnalog.cpp @@ -1,5 +1,6 @@ #include "ZigbeeAnalog.h" #if CONFIG_ZB_ENABLED +#include ZigbeeAnalog::ZigbeeAnalog(uint8_t endpoint) : ZigbeeEP(endpoint) { _device_id = ESP_ZB_HA_SIMPLE_SENSOR_DEVICE_ID; @@ -20,6 +21,8 @@ bool ZigbeeAnalog::addAnalogInput() { "Analog Input"; uint32_t application_type = 0x00000000 | (ESP_ZB_ZCL_AI_GROUP_ID << 24); float resolution = 0.1; // Default resolution of 0.1 + float min = -FLT_MAX; // Default min value for float + float max = FLT_MAX; // Default max value for float esp_err_t ret = esp_zb_analog_input_cluster_add_attr(esp_zb_analog_input_cluster, ESP_ZB_ZCL_ATTR_ANALOG_INPUT_DESCRIPTION_ID, (void *)default_description); if (ret != ESP_OK) { @@ -39,11 +42,24 @@ bool ZigbeeAnalog::addAnalogInput() { return false; } + ret = esp_zb_analog_input_cluster_add_attr(esp_zb_analog_input_cluster, ESP_ZB_ZCL_ATTR_ANALOG_INPUT_MIN_PRESENT_VALUE_ID, (void *)&min); + if (ret != ESP_OK) { + log_e("Failed to set min value: 0x%x: %s", ret, esp_err_to_name(ret)); + return false; + } + + ret = esp_zb_analog_input_cluster_add_attr(esp_zb_analog_input_cluster, ESP_ZB_ZCL_ATTR_ANALOG_INPUT_MAX_PRESENT_VALUE_ID, (void *)&max); + if (ret != ESP_OK) { + log_e("Failed to set max value: 0x%x: %s", ret, esp_err_to_name(ret)); + return false; + } + ret = esp_zb_cluster_list_add_analog_input_cluster(_cluster_list, esp_zb_analog_input_cluster, ESP_ZB_ZCL_CLUSTER_SERVER_ROLE); if (ret != ESP_OK) { log_e("Failed to add Analog Input cluster: 0x%x: %s", ret, esp_err_to_name(ret)); return false; } + _analog_clusters |= ANALOG_INPUT; return true; } @@ -76,6 +92,8 @@ bool ZigbeeAnalog::addAnalogOutput() { "Analog Output"; uint32_t application_type = 0x00000000 | (ESP_ZB_ZCL_AO_GROUP_ID << 24); float resolution = 1; // Default resolution of 1 + float min = -FLT_MAX; // Default min value for float + float max = FLT_MAX; // Default max value for float esp_err_t ret = esp_zb_analog_output_cluster_add_attr(esp_zb_analog_output_cluster, ESP_ZB_ZCL_ATTR_ANALOG_OUTPUT_DESCRIPTION_ID, (void *)default_description); @@ -96,6 +114,18 @@ bool ZigbeeAnalog::addAnalogOutput() { return false; } + ret = esp_zb_analog_output_cluster_add_attr(esp_zb_analog_output_cluster, ESP_ZB_ZCL_ATTR_ANALOG_OUTPUT_MIN_PRESENT_VALUE_ID, (void *)&min); + if (ret != ESP_OK) { + log_e("Failed to set min value: 0x%x: %s", ret, esp_err_to_name(ret)); + return false; + } + + ret = esp_zb_analog_output_cluster_add_attr(esp_zb_analog_output_cluster, ESP_ZB_ZCL_ATTR_ANALOG_OUTPUT_MAX_PRESENT_VALUE_ID, (void *)&max); + if (ret != ESP_OK) { + log_e("Failed to set max value: 0x%x: %s", ret, esp_err_to_name(ret)); + return false; + } + ret = esp_zb_cluster_list_add_analog_output_cluster(_cluster_list, esp_zb_analog_output_cluster, ESP_ZB_ZCL_CLUSTER_SERVER_ROLE); if (ret != ESP_OK) { log_e("Failed to add Analog Output cluster: 0x%x: %s", ret, esp_err_to_name(ret)); @@ -376,4 +406,58 @@ bool ZigbeeAnalog::setAnalogOutputResolution(float resolution) { return true; } +bool ZigbeeAnalog::setAnalogOutputMinMax(float min, float max) { + if (!(_analog_clusters & ANALOG_OUTPUT)) { + log_e("Analog Output cluster not added"); + return false; + } + + esp_zb_attribute_list_t *analog_output_cluster = + esp_zb_cluster_list_get_cluster(_cluster_list, ESP_ZB_ZCL_CLUSTER_ID_ANALOG_OUTPUT, ESP_ZB_ZCL_CLUSTER_SERVER_ROLE); + if (analog_output_cluster == nullptr) { + log_e("Failed to get analog output cluster"); + return false; + } + + esp_err_t ret = esp_zb_cluster_update_attr(analog_output_cluster, ESP_ZB_ZCL_ATTR_ANALOG_OUTPUT_MIN_PRESENT_VALUE_ID, (void *)&min); + if (ret != ESP_OK) { + log_e("Failed to set min value: 0x%x: %s", ret, esp_err_to_name(ret)); + return false; + } + + ret = esp_zb_cluster_update_attr(analog_output_cluster, ESP_ZB_ZCL_ATTR_ANALOG_OUTPUT_MAX_PRESENT_VALUE_ID, (void *)&max); + if (ret != ESP_OK) { + log_e("Failed to set max value: 0x%x: %s", ret, esp_err_to_name(ret)); + return false; + } + return true; +} + +bool ZigbeeAnalog::setAnalogInputMinMax(float min, float max) { + if (!(_analog_clusters & ANALOG_INPUT)) { + log_e("Analog Input cluster not added"); + return false; + } + + esp_zb_attribute_list_t *analog_input_cluster = + esp_zb_cluster_list_get_cluster(_cluster_list, ESP_ZB_ZCL_CLUSTER_ID_ANALOG_INPUT, ESP_ZB_ZCL_CLUSTER_SERVER_ROLE); + if (analog_input_cluster == nullptr) { + log_e("Failed to get analog input cluster"); + return false; + } + + esp_err_t ret = esp_zb_cluster_update_attr(analog_input_cluster, ESP_ZB_ZCL_ATTR_ANALOG_INPUT_MIN_PRESENT_VALUE_ID, (void *)&min); + if (ret != ESP_OK) { + log_e("Failed to set min value: 0x%x: %s", ret, esp_err_to_name(ret)); + return false; + } + + ret = esp_zb_cluster_update_attr(analog_input_cluster, ESP_ZB_ZCL_ATTR_ANALOG_INPUT_MAX_PRESENT_VALUE_ID, (void *)&max); + if (ret != ESP_OK) { + log_e("Failed to set max value: 0x%x: %s", ret, esp_err_to_name(ret)); + return false; + } + return true; +} + #endif // CONFIG_ZB_ENABLED diff --git a/libraries/Zigbee/src/ep/ZigbeeAnalog.h b/libraries/Zigbee/src/ep/ZigbeeAnalog.h index bbc5f6d6fc2..5218a0b7d60 100644 --- a/libraries/Zigbee/src/ep/ZigbeeAnalog.h +++ b/libraries/Zigbee/src/ep/ZigbeeAnalog.h @@ -41,6 +41,10 @@ class ZigbeeAnalog : public ZigbeeEP { bool setAnalogOutputDescription(const char *description); bool setAnalogOutputResolution(float resolution); + // Set the min and max values for the analog Input/Output + bool setAnalogOutputMinMax(float min, float max); + bool setAnalogInputMinMax(float min, float max); + // Use to set a cb function to be called on analog output change void onAnalogOutputChange(void (*callback)(float analog)) { _on_analog_output_change = callback; From d6a76da0a5134bce8dfd4961af7e9c828bf21328 Mon Sep 17 00:00:00 2001 From: Niki Waibel Date: Tue, 10 Jun 2025 10:55:43 +0200 Subject: [PATCH 044/173] fix(libraries/asyncudp): IPv4 ONLY listenMulticast() (#11444) AsyncUDP::listenMulticast() properly receives packets sent to IPv4 multicast addresses like 239.1.2.3, but it is not receiving packets sent to IPv6 multicast addresses like ff12::6ood:cafe. The root cause is a bit hidden: listen(NULL, port) would match AsyncUDP::listen(const ip_addr_t *addr, uint16_t port), which calls _udp_bind(_pcb, addr, port), which uses the lwIP API to call udp_bind(struct udp_pcb *pcb, const ip_addr_t *ipaddr, u16_t port) at the end. If lwIP has LWIP_IPV4 enabled, it checks if ipaddr == NULL and sets it to IP4_ADDR_ANY. So an IPv6 address is never bound. This fix checks the IP address passed to AsyncUDP::listenMulticast(). If it is an IPv6 address, it constructs and passes the IPv6 any address (::); otherwise (IPv4), it constructs and passes the IPv4 any address (0.0.0.0). --- libraries/AsyncUDP/src/AsyncUDP.cpp | 15 ++++++++++++++- 1 file changed, 14 insertions(+), 1 deletion(-) diff --git a/libraries/AsyncUDP/src/AsyncUDP.cpp b/libraries/AsyncUDP/src/AsyncUDP.cpp index f44cc839c97..cab9c951921 100644 --- a/libraries/AsyncUDP/src/AsyncUDP.cpp +++ b/libraries/AsyncUDP/src/AsyncUDP.cpp @@ -682,6 +682,8 @@ static esp_err_t joinMulticastGroup(const ip_addr_t *addr, bool join, tcpip_adap } bool AsyncUDP::listenMulticast(const ip_addr_t *addr, uint16_t port, uint8_t ttl, tcpip_adapter_if_t tcpip_if) { + ip_addr_t bind_addr; + if (!ip_addr_ismulticast(addr)) { return false; } @@ -690,7 +692,18 @@ bool AsyncUDP::listenMulticast(const ip_addr_t *addr, uint16_t port, uint8_t ttl return false; } - if (!listen(NULL, port)) { +#if CONFIG_LWIP_IPV6 + if (IP_IS_V6(addr)) { + IP_SET_TYPE(&bind_addr, IPADDR_TYPE_V6); + ip6_addr_set_any(&bind_addr.u_addr.ip6); + } else { +#endif + IP_SET_TYPE(&bind_addr, IPADDR_TYPE_V4); + ip4_addr_set_any(&bind_addr.u_addr.ip4); +#if CONFIG_LWIP_IPV6 + } +#endif + if (!listen(&bind_addr, port)) { return false; } From cbdaee6f52b78fa747693dbb718c2cafe99015b0 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jan=20Proch=C3=A1zka?= <90197375+P-R-O-C-H-Y@users.noreply.github.com> Date: Tue, 10 Jun 2025 11:57:07 +0200 Subject: [PATCH 045/173] feat(ledc): Improve timer management with frequency/resolution matching (#11452) * feat(ledc): Improve timer management with frequency/resolution matching * fix(ci): Fix uninitialized timer variable warning * Update cores/esp32/esp32-hal-ledc.c Co-authored-by: Lucas Saavedra Vaz <32426024+lucasssvaz@users.noreply.github.com> * ci(pre-commit): Apply automatic fixes --------- Co-authored-by: Lucas Saavedra Vaz <32426024+lucasssvaz@users.noreply.github.com> Co-authored-by: pre-commit-ci-lite[bot] <117423508+pre-commit-ci-lite[bot]@users.noreply.github.com> --- cores/esp32/esp32-hal-ledc.c | 124 +++++++++++++++++++++++++++++++---- cores/esp32/esp32-hal-ledc.h | 2 + 2 files changed, 115 insertions(+), 11 deletions(-) diff --git a/cores/esp32/esp32-hal-ledc.c b/cores/esp32/esp32-hal-ledc.c index 039fa1312f1..764c2803b4b 100644 --- a/cores/esp32/esp32-hal-ledc.c +++ b/cores/esp32/esp32-hal-ledc.c @@ -45,6 +45,93 @@ typedef struct { ledc_periph_t ledc_handle = {0}; +// Helper function to find a timer with matching frequency and resolution +static bool find_matching_timer(uint8_t speed_mode, uint32_t freq, uint8_t resolution, uint8_t *timer_num) { + log_d("Searching for timer with freq=%u, resolution=%u", freq, resolution); + // Check all channels to find one with matching frequency and resolution + for (uint8_t i = 0; i < SOC_GPIO_PIN_COUNT; i++) { + if (!perimanPinIsValid(i)) { + continue; + } + peripheral_bus_type_t type = perimanGetPinBusType(i); + if (type == ESP32_BUS_TYPE_LEDC) { + ledc_channel_handle_t *bus = (ledc_channel_handle_t *)perimanGetPinBus(i, ESP32_BUS_TYPE_LEDC); + if (bus != NULL && (bus->channel / 8) == speed_mode && bus->freq_hz == freq && bus->channel_resolution == resolution) { + log_d("Found matching timer %u for freq=%u, resolution=%u", bus->timer_num, freq, resolution); + *timer_num = bus->timer_num; + return true; + } + } + } + log_d("No matching timer found for freq=%u, resolution=%u", freq, resolution); + return false; +} + +// Helper function to find an unused timer +static bool find_free_timer(uint8_t speed_mode, uint8_t *timer_num) { + // Check which timers are in use + uint8_t used_timers = 0; + for (uint8_t i = 0; i < SOC_GPIO_PIN_COUNT; i++) { + if (!perimanPinIsValid(i)) { + continue; + } + peripheral_bus_type_t type = perimanGetPinBusType(i); + if (type == ESP32_BUS_TYPE_LEDC) { + ledc_channel_handle_t *bus = (ledc_channel_handle_t *)perimanGetPinBus(i, ESP32_BUS_TYPE_LEDC); + if (bus != NULL && (bus->channel / 8) == speed_mode) { + log_d("Timer %u is in use by channel %u", bus->timer_num, bus->channel); + used_timers |= (1 << bus->timer_num); + } + } + } + + // Find first unused timer + for (uint8_t i = 0; i < SOC_LEDC_TIMER_NUM; i++) { + if (!(used_timers & (1 << i))) { + log_d("Found free timer %u", i); + *timer_num = i; + return true; + } + } + log_e("No free timers available"); + return false; +} + +// Helper function to remove a channel from a timer and clear timer if no channels are using it +static void remove_channel_from_timer(uint8_t speed_mode, uint8_t timer_num, uint8_t channel) { + log_d("Removing channel %u from timer %u in speed_mode %u", channel, timer_num, speed_mode); + + // Check if any other channels are using this timer + bool timer_in_use = false; + for (uint8_t i = 0; i < SOC_GPIO_PIN_COUNT; i++) { + if (!perimanPinIsValid(i)) { + continue; + } + peripheral_bus_type_t type = perimanGetPinBusType(i); + if (type == ESP32_BUS_TYPE_LEDC) { + ledc_channel_handle_t *bus = (ledc_channel_handle_t *)perimanGetPinBus(i, ESP32_BUS_TYPE_LEDC); + if (bus != NULL && (bus->channel / 8) == speed_mode && bus->timer_num == timer_num && bus->channel != channel) { + log_d("Timer %u is still in use by channel %u", timer_num, bus->channel); + timer_in_use = true; + break; + } + } + } + + if (!timer_in_use) { + log_d("No other channels using timer %u, deconfiguring timer", timer_num); + // Stop the timer + ledc_timer_pause(speed_mode, timer_num); + // Deconfigure the timer + ledc_timer_config_t ledc_timer; + memset((void *)&ledc_timer, 0, sizeof(ledc_timer_config_t)); + ledc_timer.speed_mode = speed_mode; + ledc_timer.timer_num = timer_num; + ledc_timer.deconfigure = true; + ledc_timer_config(&ledc_timer); + } +} + static bool fade_initialized = false; static ledc_clk_cfg_t clock_source = LEDC_DEFAULT_CLK; @@ -81,6 +168,8 @@ static bool ledcDetachBus(void *bus) { } pinMatrixOutDetach(handle->pin, false, false); if (!channel_found) { + uint8_t group = (handle->channel / 8); + remove_channel_from_timer(group, handle->timer_num, handle->channel % 8); ledc_handle.used_channels &= ~(1UL << handle->channel); } free(handle); @@ -117,8 +206,10 @@ bool ledcAttachChannel(uint8_t pin, uint32_t freq, uint8_t resolution, uint8_t c return false; } - uint8_t group = (channel / 8), timer = ((channel / 2) % 4); + uint8_t group = (channel / 8); + uint8_t timer = 0; bool channel_used = ledc_handle.used_channels & (1UL << channel); + if (channel_used) { log_i("Channel %u is already set up, given frequency and resolution will be ignored", channel); if (ledc_set_pin(pin, group, channel % 8) != ESP_OK) { @@ -126,17 +217,26 @@ bool ledcAttachChannel(uint8_t pin, uint32_t freq, uint8_t resolution, uint8_t c return false; } } else { - ledc_timer_config_t ledc_timer; - memset((void *)&ledc_timer, 0, sizeof(ledc_timer_config_t)); - ledc_timer.speed_mode = group; - ledc_timer.timer_num = timer; - ledc_timer.duty_resolution = resolution; - ledc_timer.freq_hz = freq; - ledc_timer.clk_cfg = clock_source; + // Find a timer with matching frequency and resolution, or a free timer + if (!find_matching_timer(group, freq, resolution, &timer)) { + if (!find_free_timer(group, &timer)) { + log_e("No free timers available for speed mode %u", group); + return false; + } - if (ledc_timer_config(&ledc_timer) != ESP_OK) { - log_e("ledc setup failed!"); - return false; + // Configure the timer if we're using a new one + ledc_timer_config_t ledc_timer; + memset((void *)&ledc_timer, 0, sizeof(ledc_timer_config_t)); + ledc_timer.speed_mode = group; + ledc_timer.timer_num = timer; + ledc_timer.duty_resolution = resolution; + ledc_timer.freq_hz = freq; + ledc_timer.clk_cfg = clock_source; + + if (ledc_timer_config(&ledc_timer) != ESP_OK) { + log_e("ledc setup failed!"); + return false; + } } uint32_t duty = ledc_get_duty(group, (channel % 8)); @@ -157,6 +257,8 @@ bool ledcAttachChannel(uint8_t pin, uint32_t freq, uint8_t resolution, uint8_t c ledc_channel_handle_t *handle = (ledc_channel_handle_t *)malloc(sizeof(ledc_channel_handle_t)); handle->pin = pin; handle->channel = channel; + handle->timer_num = timer; + handle->freq_hz = freq; #ifndef SOC_LEDC_SUPPORT_FADE_STOP handle->lock = NULL; #endif diff --git a/cores/esp32/esp32-hal-ledc.h b/cores/esp32/esp32-hal-ledc.h index 5b44aaad452..f1a27dd4f7a 100644 --- a/cores/esp32/esp32-hal-ledc.h +++ b/cores/esp32/esp32-hal-ledc.h @@ -51,6 +51,8 @@ typedef struct { uint8_t pin; // Pin assigned to channel uint8_t channel; // Channel number uint8_t channel_resolution; // Resolution of channel + uint8_t timer_num; // Timer number used by this channel + uint32_t freq_hz; // Frequency configured for this channel voidFuncPtr fn; void *arg; #ifndef SOC_LEDC_SUPPORT_FADE_STOP From c21ef70a156aa0071f00ee6d7a00e0be2d0aa17e Mon Sep 17 00:00:00 2001 From: Lucas Saavedra Vaz <32426024+lucasssvaz@users.noreply.github.com> Date: Tue, 10 Jun 2025 07:09:27 -0300 Subject: [PATCH 046/173] fix(release): Replace all assets with chinese mirrors (#11323) * fix(release): Replace all assets with chinese mirrors * feat(release): Add script to append "-cn" to versions * docs(install): Add instructions for users in China --- .github/scripts/on-release.sh | 10 +++-- .github/scripts/release_append_cn.py | 56 ++++++++++++++++++++++++++++ docs/en/installing.rst | 2 + 3 files changed, 64 insertions(+), 4 deletions(-) create mode 100755 .github/scripts/release_append_cn.py diff --git a/.github/scripts/on-release.sh b/.github/scripts/on-release.sh index dafbf3d6a1c..275c74f8ea5 100755 --- a/.github/scripts/on-release.sh +++ b/.github/scripts/on-release.sh @@ -342,12 +342,14 @@ jq_arg=".packages[0].platforms[0].version = \"$RELEASE_TAG\" | \ echo "Generating $PACKAGE_JSON_DEV ..." cat "$PACKAGE_JSON_TEMPLATE" | jq "$jq_arg" > "$OUTPUT_DIR/$PACKAGE_JSON_DEV" # On MacOS the sed command won't skip the first match. Use gsed instead. -sed '0,/github\.com\/espressif\//!s|github\.com/espressif/|dl.espressif.cn/github_assets/espressif/|g' "$OUTPUT_DIR/$PACKAGE_JSON_DEV" > "$OUTPUT_DIR/$PACKAGE_JSON_DEV_CN" +sed '0,/github\.com\//!s|github\.com/|dl.espressif.cn/github_assets/|g' "$OUTPUT_DIR/$PACKAGE_JSON_DEV" > "$OUTPUT_DIR/$PACKAGE_JSON_DEV_CN" +python "$SCRIPTS_DIR/release_append_cn.py" "$OUTPUT_DIR/$PACKAGE_JSON_DEV_CN" if [ "$RELEASE_PRE" == "false" ]; then echo "Generating $PACKAGE_JSON_REL ..." cat "$PACKAGE_JSON_TEMPLATE" | jq "$jq_arg" > "$OUTPUT_DIR/$PACKAGE_JSON_REL" # On MacOS the sed command won't skip the first match. Use gsed instead. - sed '0,/github\.com\/espressif\//!s|github\.com/espressif/|dl.espressif.cn/github_assets/espressif/|g' "$OUTPUT_DIR/$PACKAGE_JSON_REL" > "$OUTPUT_DIR/$PACKAGE_JSON_REL_CN" + sed '0,/github\.com\//!s|github\.com/|dl.espressif.cn/github_assets/|g' "$OUTPUT_DIR/$PACKAGE_JSON_REL" > "$OUTPUT_DIR/$PACKAGE_JSON_REL_CN" + python "$SCRIPTS_DIR/release_append_cn.py" "$OUTPUT_DIR/$PACKAGE_JSON_REL_CN" fi # Figure out the last release or pre-release @@ -456,14 +458,14 @@ echo "Uploading $PACKAGE_JSON_DEV ..." echo "Download URL: $(git_safe_upload_asset "$OUTPUT_DIR/$PACKAGE_JSON_DEV")" echo "Pages URL: $(git_safe_upload_to_pages "$PACKAGE_JSON_DEV" "$OUTPUT_DIR/$PACKAGE_JSON_DEV")" echo "Download CN URL: $(git_safe_upload_asset "$OUTPUT_DIR/$PACKAGE_JSON_DEV_CN")" -echo "Pages CN URL: $(git_safe_upload_to_pages "$PACKAGE_JSON_DEV" "$OUTPUT_DIR/$PACKAGE_JSON_DEV_CN")" +echo "Pages CN URL: $(git_safe_upload_to_pages "$PACKAGE_JSON_DEV_CN" "$OUTPUT_DIR/$PACKAGE_JSON_DEV_CN")" echo if [ "$RELEASE_PRE" == "false" ]; then echo "Uploading $PACKAGE_JSON_REL ..." echo "Download URL: $(git_safe_upload_asset "$OUTPUT_DIR/$PACKAGE_JSON_REL")" echo "Pages URL: $(git_safe_upload_to_pages "$PACKAGE_JSON_REL" "$OUTPUT_DIR/$PACKAGE_JSON_REL")" echo "Download CN URL: $(git_safe_upload_asset "$OUTPUT_DIR/$PACKAGE_JSON_REL_CN")" - echo "Pages CN URL: $(git_safe_upload_to_pages "$PACKAGE_JSON_REL" "$OUTPUT_DIR/$PACKAGE_JSON_REL_CN")" + echo "Pages CN URL: $(git_safe_upload_to_pages "$PACKAGE_JSON_REL_CN" "$OUTPUT_DIR/$PACKAGE_JSON_REL_CN")" echo fi diff --git a/.github/scripts/release_append_cn.py b/.github/scripts/release_append_cn.py new file mode 100755 index 00000000000..b29fe0c31ba --- /dev/null +++ b/.github/scripts/release_append_cn.py @@ -0,0 +1,56 @@ + +#!/usr/bin/env python3 + +# Arduino IDE provides by default a package file for the ESP32. This causes version conflicts +# when the user tries to use the JSON file with the Chinese mirrors. +# +# The downside is that the Arduino IDE will always warn the user that updates are available as it +# will consider the version from the Chinese mirrors as a pre-release version. +# +# This script is used to append "-cn" to all versions in the package_esp32_index_cn.json file so that +# the user can select the Chinese mirrors without conflicts. +# +# If Arduino ever stops providing the package_esp32_index.json file by default, +# this script can be removed and the tags reverted. + +import json + +def append_cn_to_versions(obj): + if isinstance(obj, dict): + # dfu-util comes from arduino.cc and not from the Chinese mirrors, so we skip it + if obj.get("name") == "dfu-util": + return + + for key, value in obj.items(): + if key == "version" and isinstance(value, str): + if not value.endswith("-cn"): + obj[key] = value + "-cn" + else: + append_cn_to_versions(value) + + elif isinstance(obj, list): + for item in obj: + append_cn_to_versions(item) + +def process_json_file(input_path, output_path=None): + with open(input_path, "r", encoding="utf-8") as f: + data = json.load(f) + + append_cn_to_versions(data) + + if output_path is None: + output_path = input_path + + with open(output_path, "w", encoding="utf-8") as f: + json.dump(data, f, indent=2) + + print(f"Updated JSON written to {output_path}") + +if __name__ == "__main__": + import sys + if len(sys.argv) < 2: + print("Usage: python release_append_cn.py input.json [output.json]") + else: + input_file = sys.argv[1] + output_file = sys.argv[2] if len(sys.argv) > 2 else None + process_json_file(input_file, output_file) diff --git a/docs/en/installing.rst b/docs/en/installing.rst index 35342020864..3ca0881c398 100644 --- a/docs/en/installing.rst +++ b/docs/en/installing.rst @@ -70,6 +70,8 @@ To start the installation process using the Boards Manager, follow these steps: :figclass: align-center - Open Boards Manager from Tools > Board menu and install *esp32* platform (and do not forget to select your ESP32 board from Tools > Board menu after installation). + Users in China must select the package version with the "-cn" suffix and perform updates manually. + Automatic updates are not supported in this region, as they target the default package without the "-cn" suffix, resulting in download failures. .. figure:: ../_static/install_guide_boards_manager_esp32.png :align: center From d71135e2ca72cef49ea417d00ae821b35cf95da3 Mon Sep 17 00:00:00 2001 From: whatsABetterNick Date: Tue, 10 Jun 2025 18:27:47 +0800 Subject: [PATCH 047/173] Fix(I2S example): make fix to the ESP32 I2S simple tone example (#10954) * made some fix to the ESP32 I2S simple tone example * edit the I2S - simple tone example * edit the I2S - simple tone example * some edit * edit comment * edit * edit * ci(pre-commit): Apply automatic fixes --------- Co-authored-by: pre-commit-ci-lite[bot] <117423508+pre-commit-ci-lite[bot]@users.noreply.github.com> --- .../examples/Simple_tone/Simple_tone.ino | 22 +++++++++++++++---- 1 file changed, 18 insertions(+), 4 deletions(-) diff --git a/libraries/ESP_I2S/examples/Simple_tone/Simple_tone.ino b/libraries/ESP_I2S/examples/Simple_tone/Simple_tone.ino index 935aa4bc50f..bba7d4f4d9d 100644 --- a/libraries/ESP_I2S/examples/Simple_tone/Simple_tone.ino +++ b/libraries/ESP_I2S/examples/Simple_tone/Simple_tone.ino @@ -24,10 +24,17 @@ 2nd September 2021 Lucas Saavedra Vaz (lucasssvaz) 22nd December 2023 + anon + 10nd February 2025 */ #include +// The GPIO pins are not fixed, most other pins could be used for the I2S function. +#define I2S_LRC 25 +#define I2S_BCLK 5 +#define I2S_DIN 26 + const int frequency = 440; // frequency of square wave in Hz const int amplitude = 500; // amplitude of square wave const int sampleRate = 8000; // sample rate in Hz @@ -36,10 +43,10 @@ i2s_data_bit_width_t bps = I2S_DATA_BIT_WIDTH_16BIT; i2s_mode_t mode = I2S_MODE_STD; i2s_slot_mode_t slot = I2S_SLOT_MODE_STEREO; -const int halfWavelength = (sampleRate / frequency); // half wavelength of square wave +const unsigned int halfWavelength = sampleRate / frequency / 2; // half wavelength of square wave int32_t sample = amplitude; // current sample value -int count = 0; +unsigned int count = 0; I2SClass i2s; @@ -47,6 +54,8 @@ void setup() { Serial.begin(115200); Serial.println("I2S simple tone"); + i2s.setPins(I2S_BCLK, I2S_LRC, I2S_DIN); + // start I2S at the sample rate with 16-bits per sample if (!i2s.begin(mode, sampleRate, bps, slot)) { Serial.println("Failed to initialize I2S!"); @@ -60,8 +69,13 @@ void loop() { sample = -1 * sample; } - i2s.write(sample); // Right channel - i2s.write(sample); // Left channel + // Left channel, the low 8 bits then high 8 bits + i2s.write(sample); + i2s.write(sample >> 8); + + // Right channel, the low 8 bits then high 8 bits + i2s.write(sample); + i2s.write(sample >> 8); // increment the counter for the next sample count++; From 422e52684b824a3fde85989ea90abf6624f2b0eb Mon Sep 17 00:00:00 2001 From: Paula Scharf <48286621+PaulaScharf@users.noreply.github.com> Date: Tue, 10 Jun 2025 13:29:27 +0200 Subject: [PATCH 048/173] fix(msc): remove weak function declaration of tud_msc_is_writable_cb (#11353) Co-authored-by: Lucas Saavedra Vaz <32426024+lucasssvaz@users.noreply.github.com> --- cores/esp32/esp32-hal-tinyusb.c | 3 --- 1 file changed, 3 deletions(-) diff --git a/cores/esp32/esp32-hal-tinyusb.c b/cores/esp32/esp32-hal-tinyusb.c index 0991e08d27f..30a827baa01 100644 --- a/cores/esp32/esp32-hal-tinyusb.c +++ b/cores/esp32/esp32-hal-tinyusb.c @@ -466,9 +466,6 @@ __attribute__((weak)) int32_t tud_msc_write10_cb(uint8_t lun, uint32_t lba, uint __attribute__((weak)) int32_t tud_msc_scsi_cb(uint8_t lun, uint8_t const scsi_cmd[16], void *buffer, uint16_t bufsize) { return -1; } -__attribute__((weak)) bool tud_msc_is_writable_cb(uint8_t lun) { - return false; -} #endif #if CFG_TUD_NCM __attribute__((weak)) bool tud_network_recv_cb(const uint8_t *src, uint16_t size) { From 228393708d49f20f9c83dfbc0bd51300464d873d Mon Sep 17 00:00:00 2001 From: SooDragon <82627949+SooDragon@users.noreply.github.com> Date: Mon, 16 Jun 2025 17:12:33 +0900 Subject: [PATCH 049/173] fix: Update Pin compatability (#11473) fix: Update Pin compatability --- variants/Geekble_ESP32C3/pins_arduino.h | 7 +++++++ 1 file changed, 7 insertions(+) diff --git a/variants/Geekble_ESP32C3/pins_arduino.h b/variants/Geekble_ESP32C3/pins_arduino.h index 660313ce849..56bda115a7e 100644 --- a/variants/Geekble_ESP32C3/pins_arduino.h +++ b/variants/Geekble_ESP32C3/pins_arduino.h @@ -22,6 +22,13 @@ static const uint8_t MOSI = 6; static const uint8_t MISO = 5; static const uint8_t SCK = 4; +static const uint8_t D5 = 5; +static const uint8_t D6 = 6; +static const uint8_t D7 = 7; +static const uint8_t D8 = 8; +static const uint8_t D9 = 9; +static const uint8_t D10 = 10; + static const uint8_t A0 = 0; static const uint8_t A1 = 1; static const uint8_t A2 = 2; From ef995b6564612f300ed1f32627aa78a1eb65d614 Mon Sep 17 00:00:00 2001 From: Sugar Glider Date: Mon, 16 Jun 2025 07:15:54 -0300 Subject: [PATCH 050/173] feat(openthread): adds native api (#11474) * feat(openthread): adds native api * feat(openthread): adds source code to CMakeLists.txt * ci(pre-commit): Apply automatic fixes --------- Co-authored-by: pre-commit-ci-lite[bot] <117423508+pre-commit-ci-lite[bot]@users.noreply.github.com> --- CMakeLists.txt | 1 + libraries/OpenThread/README.md | 201 +++++++++- .../examples/{ => CLI}/COAP/coap_lamp/ci.json | 0 .../{ => CLI}/COAP/coap_lamp/coap_lamp.ino | 9 +- .../{ => CLI}/COAP/coap_switch/ci.json | 0 .../COAP/coap_switch/coap_switch.ino | 9 +- .../{ => CLI}/SimpleCLI/SimpleCLI.ino | 5 +- .../examples/{ => CLI}/SimpleCLI/ci.json | 0 .../{ => CLI}/SimpleNode/SimpleNode.ino | 9 +- .../examples/{ => CLI}/SimpleNode/ci.json | 0 .../ExtendedRouterNode/ExtendedRouterNode.ino | 9 +- .../ExtendedRouterNode/ci.json | 0 .../LeaderNode/LeaderNode.ino | 9 +- .../SimpleThreadNetwork/LeaderNode/ci.json | 0 .../RouterNode/RouterNode.ino | 9 +- .../SimpleThreadNetwork/RouterNode/ci.json | 0 .../{ => CLI}/ThreadScan/ThreadScan.ino | 7 +- .../examples/{ => CLI}/ThreadScan/ci.json | 0 .../examples/{ => CLI}/onReceive/ci.json | 0 .../{ => CLI}/onReceive/onReceive.ino | 3 +- .../LeaderNode/LeaderNode.ino | 34 ++ .../SimpleThreadNetwork/LeaderNode/ci.json | 6 + .../RouterNode/RouterNode.ino | 29 ++ .../SimpleThreadNetwork/RouterNode/ci.json | 6 + libraries/OpenThread/keywords.txt | 26 ++ libraries/OpenThread/src/OThread.cpp | 364 ++++++++++++++++++ libraries/OpenThread/src/OThread.h | 107 +++++ libraries/OpenThread/src/OThreadCLI.cpp | 156 ++------ libraries/OpenThread/src/OThreadCLI.h | 6 +- libraries/OpenThread/src/OThreadCLI_Util.cpp | 22 +- libraries/OpenThread/src/OThreadCLI_Util.h | 11 - 31 files changed, 833 insertions(+), 205 deletions(-) rename libraries/OpenThread/examples/{ => CLI}/COAP/coap_lamp/ci.json (100%) rename libraries/OpenThread/examples/{ => CLI}/COAP/coap_lamp/coap_lamp.ino (95%) rename libraries/OpenThread/examples/{ => CLI}/COAP/coap_switch/ci.json (100%) rename libraries/OpenThread/examples/{ => CLI}/COAP/coap_switch/coap_switch.ino (95%) rename libraries/OpenThread/examples/{ => CLI}/SimpleCLI/SimpleCLI.ino (89%) rename libraries/OpenThread/examples/{ => CLI}/SimpleCLI/ci.json (100%) rename libraries/OpenThread/examples/{ => CLI}/SimpleNode/SimpleNode.ino (82%) rename libraries/OpenThread/examples/{ => CLI}/SimpleNode/ci.json (100%) rename libraries/OpenThread/examples/{ => CLI}/SimpleThreadNetwork/ExtendedRouterNode/ExtendedRouterNode.ino (89%) rename libraries/OpenThread/examples/{ => CLI}/SimpleThreadNetwork/ExtendedRouterNode/ci.json (100%) rename libraries/OpenThread/examples/{ => CLI}/SimpleThreadNetwork/LeaderNode/LeaderNode.ino (93%) rename libraries/OpenThread/examples/{ => CLI}/SimpleThreadNetwork/LeaderNode/ci.json (100%) rename libraries/OpenThread/examples/{ => CLI}/SimpleThreadNetwork/RouterNode/RouterNode.ino (92%) rename libraries/OpenThread/examples/{ => CLI}/SimpleThreadNetwork/RouterNode/ci.json (100%) rename libraries/OpenThread/examples/{ => CLI}/ThreadScan/ThreadScan.ino (89%) rename libraries/OpenThread/examples/{ => CLI}/ThreadScan/ci.json (100%) rename libraries/OpenThread/examples/{ => CLI}/onReceive/ci.json (100%) rename libraries/OpenThread/examples/{ => CLI}/onReceive/onReceive.ino (96%) create mode 100644 libraries/OpenThread/examples/Native/SimpleThreadNetwork/LeaderNode/LeaderNode.ino create mode 100644 libraries/OpenThread/examples/Native/SimpleThreadNetwork/LeaderNode/ci.json create mode 100644 libraries/OpenThread/examples/Native/SimpleThreadNetwork/RouterNode/RouterNode.ino create mode 100644 libraries/OpenThread/examples/Native/SimpleThreadNetwork/RouterNode/ci.json create mode 100644 libraries/OpenThread/src/OThread.cpp create mode 100644 libraries/OpenThread/src/OThread.h diff --git a/CMakeLists.txt b/CMakeLists.txt index d53d612962d..e8f44ac5ee0 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -165,6 +165,7 @@ set(ARDUINO_LIBRARY_LittleFS_SRCS libraries/LittleFS/src/LittleFS.cpp) set(ARDUINO_LIBRARY_NetBIOS_SRCS libraries/NetBIOS/src/NetBIOS.cpp) set(ARDUINO_LIBRARY_OpenThread_SRCS + libraries/OpenThread/src/OThread.cpp libraries/OpenThread/src/OThreadCLI.cpp libraries/OpenThread/src/OThreadCLI_Util.cpp) diff --git a/libraries/OpenThread/README.md b/libraries/OpenThread/README.md index cd9deb9ebf6..8d0386f34fb 100644 --- a/libraries/OpenThread/README.md +++ b/libraries/OpenThread/README.md @@ -1,9 +1,177 @@ | Supported Targets | ESP32-C6 | ESP32-H2 | | ----------------- | -------- | -------- | -# ESP32 Arduino OpenThreadCLI +# General View -The `OpenThreadCLI` class is an Arduino API for interacting with the OpenThread Command Line Interface (CLI). It allows you to manage and configure the Thread stack using a command-line interface. +This Arduino OpenThread Library allows using ESP OpenThread implementation using CLI and/or Native OpenThread API. + +The Library implements 3 C++ Classes: +- `OThread` Class for Native OpenThread API +- `OThreadCLI` Class for CLI OpenThread API +- `DataSet` Class for OpenThread dataset manipulation using Native `OThread` Class + +# ESP32 Arduino OpenThread Native + +The `OThread` class provides methods for managing the OpenThread instance and controlling the Thread network. It allows you to initialize, start, stop, and manage the Thread network using native OpenThread APIs. + +## Class Definition + +```cpp +class OpenThread { + public: + static bool otStarted; // Indicates whether the OpenThread stack is running. + + // Get the current Thread device role (e.g., Leader, Router, Child, etc.). + static ot_device_role_t otGetDeviceRole(); + + // Get the current Thread device role as a string. + static const char *otGetStringDeviceRole(); + + // Print network information (e.g., network name, channel, PAN ID) to the specified stream. + static void otPrintNetworkInformation(Stream &output); + + OpenThread(); + ~OpenThread(); + + // Returns true if the OpenThread stack is running. + operator bool() const; + + // Initialize the OpenThread stack. + static void begin(bool OThreadAutoStart = true); + + // Deinitialize the OpenThread stack. + static void end(); + + // Start the Thread network. + void start(); + + // Stop the Thread network. + void stop(); + + // Bring up the Thread network interface (equivalent to "ifconfig up"). + void networkInterfaceUp(); + + // Bring down the Thread network interface (equivalent to "ifconfig down"). + void networkInterfaceDown(); + + // Commit a dataset to the OpenThread instance. + void commitDataSet(const DataSet &dataset); + +private: + static otInstance *mInstance; // Pointer to the OpenThread instance. + DataSet mCurrentDataSet; // Current dataset being used by the OpenThread instance. +}; + +extern OpenThread OThread; +``` +## Class Overview + +The `OThread` class provides a simple and intuitive interface for managing the OpenThread stack and Thread network. It abstracts the complexity of the OpenThread APIs and provides Arduino-style methods for common operations. + +## Public Methods +### Initialization and Deinitialization +- `begin(bool OThreadAutoStart = true)`: Initializes the OpenThread stack. If `OThreadAutoStart` is `true`, the Thread network will start automatically using NVS data. +- `end()`: Deinitializes the OpenThread stack and releases resources. +### Thread Network Control +- `start()`: Starts the Thread network. This is equivalent to the CLI command "thread start". +- `stop()`: Stops the Thread network. This is equivalent to the CLI command "thread stop". +### Network Interface Control +- `networkInterfaceUp()`: Brings up the Thread network interface. This is equivalent to the CLI command "ifconfig up". +- `networkInterfaceDown()`: Brings down the Thread network interface. This is equivalent to the CLI command "ifconfig down". +### Dataset Management +- `commitDataSet(const DataSet &dataset)`: Commits a dataset to the OpenThread instance. This is used to configure the Thread network with specific parameters (e.g., network name, channel, PAN ID). +### Network Information +- `otGetDeviceRole()`: Returns the current Thread device role as an `ot_device_role_t` enum (e.g., `OT_ROLE_LEADER`, `OT_ROLE_ROUTER`). +- `otGetStringDeviceRole()`: Returns the current Thread device role as a string (e.g., "Leader", "Router"). +- `otPrintNetworkInformation(Stream &output)`: Prints the current network information (e.g., network name, channel, PAN ID) to the specified stream. + +## Key Features +- **Initialization and Cleanup**: Easily initialize and deinitialize the OpenThread stack. +- **Network Control**: Start and stop the Thread network with simple method calls. +- **Dataset Management**: Configure the Thread network using the `DataSet` class and commit it to the OpenThread instance. +- **Network Information**: Retrieve and print the current network information and device role. + +## Notes +- The `OThread` class is designed to simplify the use of OpenThread APIs in Arduino sketches. +- It works seamlessly with the DataSet class for managing Thread network configurations. +- Ensure that the OpenThread stack is initialized (`OThread.begin()`) before calling other methods. + +This documentation provides a comprehensive overview of the `OThread` class, its methods, and example usage. It is designed to help developers quickly integrate OpenThread functionality into their Arduino projects. + +# DataSet Class + +The `DataSet` class provides a structured way to manage and configure Thread network datasets using native OpenThread APIs. It allows you to set and retrieve network parameters such as the network name, channel, PAN ID, and more. The `DataSet` class works seamlessly with the `OThread` class to apply these configurations to the OpenThread instance. + +## Class Definition + +```cpp +class DataSet { +public: + DataSet(); + void clear(); + void initNew(); + const otOperationalDataset &getDataset() const; + + // Setters + void setNetworkName(const char *name); + void setExtendedPanId(const uint8_t *extPanId); + void setNetworkKey(const uint8_t *key); + void setChannel(uint8_t channel); + void setPanId(uint16_t panId); + + // Getters + const char *getNetworkName() const; + const uint8_t *getExtendedPanId() const; + const uint8_t *getNetworkKey() const; + uint8_t getChannel() const; + uint16_t getPanId() const; + + // Apply the dataset to the OpenThread instance + void apply(otInstance *instance); + +private: + otOperationalDataset mDataset; // Internal representation of the dataset +}; +``` + +## Class Overview +The DataSet` class simplifies the management of Thread network datasets by providing intuitive methods for setting, retrieving, and applying network parameters. It abstracts the complexity of the OpenThread dataset APIs and provides Arduino-style methods for common operations. + +## Public Methods +### Initialization +- `DataSet()`: Constructor that initializes an empty dataset. +- `void clear()`: Clears the dataset, resetting all fields to their default values. +- `void initNew()`: Initializes a new dataset with default values (equivalent to the CLI command dataset init new). +### Setters +- `void setNetworkName(const char *name)`: Sets the network name. +- `void setExtendedPanId(const uint8_t *extPanId)`: Sets the extended PAN ID. +- `void setNetworkKey(const uint8_t *key)`: Sets the network key. +- `void setChannel(uint8_t channel)`: Sets the channel. +- `void setPanId(uint16_t panId)`: Sets the PAN ID. +### Getters +- `const char *getNetworkName() const`: Retrieves the network name. +- `const uint8_t *getExtendedPanId() const`: Retrieves the extended PAN ID. +- `const uint8_t *getNetworkKey() const`: Retrieves the network key. +- `uint8_t getChannel() const`: Retrieves the channel. +- `uint16_t getPanId() const`: Retrieves the PAN ID. +### Dataset Application +- `void apply(otInstance *instance)`: Applies the dataset to the specified OpenThread instance. + +## Key Features +- **Dataset Initialization**: Easily initialize a new dataset with default values using initNew(). +- **Custom Configuration**: Set custom network parameters such as the network name, channel, and PAN ID using setter methods. +- **Dataset Application**: Apply the configured dataset to the OpenThread instance using apply(). + +** Notes +- The `DataSet` class is designed to work seamlessly with the `OThread` class for managing Thread network configurations. +- Ensure that the OpenThread stack is initialized (`OThread.begin()`) before applying a dataset. +- The initNew()`` method provides default values for the dataset, which can be customized using the setter methods. + +This documentation provides a comprehensive overview of the `DataSet` class, its methods, and example usage. It is designed to help developers easily manage Thread network configurations in their Arduino projects. + +# OpenThreadCLI Class + +The `OpenThreadCLI` class is an Arduino API for interacting with the OpenThread Command Line Interface (CLI). It allows you to send commands to the OpenThread stack and receive responses. This class is designed to simplify the use of OpenThread CLI commands in Arduino sketches. There is one main class called `OpenThreadCLI` and a global object used to operate OpenThread CLI, called `OThreadCLI`.\ Some [helper functions](helper_functions.md) were made available for working with the OpenThread CLI environment. @@ -20,7 +188,7 @@ Below are the details of the class: class OpenThreadCLI : public Stream { private: static size_t setBuffer(QueueHandle_t &queue, size_t len); - bool otStarted = false; + static bool otCLIStarted = false; public: OpenThreadCLI(); @@ -59,14 +227,35 @@ extern OpenThreadCLI OThreadCLI; - You can customize the console behavior by adjusting parameters such as echoback and buffer sizes. ## Public Methods +### Initialization and Deinitialization +- `begin()`: Initializes the OpenThread stack (optional auto-start). +- `end()`: Deinitializes the OpenThread stack and releases resources. +### Console Management - `startConsole(Stream& otStream, bool echoback = true, const char* prompt = "ot> ")`: Starts the OpenThread console with the specified stream, echoback option, and prompt. - `stopConsole()`: Stops the OpenThread console. - `setPrompt(char* prompt)`: Changes the console prompt (set to NULL for an empty prompt). - `setEchoBack(bool echoback)`: Changes the console echoback option. - `setStream(Stream& otStream)`: Changes the console Stream object. - `onReceive(OnReceiveCb_t func)`: Sets a callback function to handle complete lines of output from the OT CLI. -- `begin(bool OThreadAutoStart = true)`: Initializes the OpenThread stack (optional auto-start). -- `end()`: Deinitializes the OpenThread stack. +### Buffer Management - `setTxBufferSize(size_t tx_queue_len)`: Sets the transmit buffer size (default is 256 bytes). - `setRxBufferSize(size_t rx_queue_len)`: Sets the receive buffer size (default is 1024 bytes). -- `write(uint8_t)`, `available()`, `read()`, `peek()`, `flush()`: Standard Stream methods implementation for OpenThread CLI object. +### Stream Methods +- `write(uint8_t)`: Writes a byte to the CLI. +- `available()`: Returns the number of bytes available to read. +- `read()`: Reads a byte from the CLI. +- `peek()`: Returns the next byte without removing it from the buffer. +- `flush()`: Flushes the CLI buffer. + +## Key Features +- **Arduino Stream Compatibility**: Inherits from the Stream class, making it compatible with Arduino's standard I/O functions. +- **Customizable Console**: Allows customization of the CLI prompt, echoback behavior, and buffer sizes. +- **Callback Support**: Provides a callback mechanism to handle CLI responses asynchronously. +- **Seamless Integration**: Designed to work seamlessly with the OThread and DataSet classes + +## Notes +- The `OThreadCLI` class is designed to simplify the use of OpenThread CLI commands in Arduino sketches. +- It works seamlessly with the `OThread` and `DataSet` classes for managing Thread networks. +- Ensure that the OpenThread stack is initialized (`OThreadCLI.begin()`) before starting the CLI console. + +This documentation provides a comprehensive overview of the `OThreadCLI` class, its methods, and example usage. It is designed to help developers easily integrate OpenThread CLI functionality into their Arduino projects. diff --git a/libraries/OpenThread/examples/COAP/coap_lamp/ci.json b/libraries/OpenThread/examples/CLI/COAP/coap_lamp/ci.json similarity index 100% rename from libraries/OpenThread/examples/COAP/coap_lamp/ci.json rename to libraries/OpenThread/examples/CLI/COAP/coap_lamp/ci.json diff --git a/libraries/OpenThread/examples/COAP/coap_lamp/coap_lamp.ino b/libraries/OpenThread/examples/CLI/COAP/coap_lamp/coap_lamp.ino similarity index 95% rename from libraries/OpenThread/examples/COAP/coap_lamp/coap_lamp.ino rename to libraries/OpenThread/examples/CLI/COAP/coap_lamp/coap_lamp.ino index 51483bb4c7c..bcaf8ae9793 100644 --- a/libraries/OpenThread/examples/COAP/coap_lamp/coap_lamp.ino +++ b/libraries/OpenThread/examples/CLI/COAP/coap_lamp/coap_lamp.ino @@ -75,18 +75,18 @@ bool otDeviceSetup(const char **otSetupCmds, uint8_t nCmds1, const char **otCoap Serial.println("OpenThread started.\r\nWaiting for activating correct Device Role."); // wait for the expected Device Role to start uint8_t tries = 24; // 24 x 2.5 sec = 1 min - while (tries && otGetDeviceRole() != expectedRole) { + while (tries && OThread.otGetDeviceRole() != expectedRole) { Serial.print("."); delay(2500); tries--; } Serial.println(); if (!tries) { - log_e("Sorry, Device Role failed by timeout! Current Role: %s.", otGetStringDeviceRole()); + log_e("Sorry, Device Role failed by timeout! Current Role: %s.", OThread.otGetStringDeviceRole()); rgbLedWrite(RGB_BUILTIN, 255, 0, 0); // RED ... failed! return false; } - Serial.printf("Device is %s.\r\n", otGetStringDeviceRole()); + Serial.printf("Device is %s.\r\n", OThread.otGetStringDeviceRole()); for (i = 0; i < nCmds2; i++) { if (!otExecCommand(otCoapCmds[i * 2], otCoapCmds[i * 2 + 1])) { break; @@ -151,7 +151,8 @@ void setup() { Serial.begin(115200); // LED starts RED, indicating not connected to Thread network. rgbLedWrite(RGB_BUILTIN, 64, 0, 0); - OThreadCLI.begin(false); // No AutoStart is necessary + OThread.begin(false); // No AutoStart is necessary + OThreadCLI.begin(); OThreadCLI.setTimeout(250); // waits 250ms for the OpenThread CLI response setupNode(); // LED goes Green when all is ready and Red when failed. diff --git a/libraries/OpenThread/examples/COAP/coap_switch/ci.json b/libraries/OpenThread/examples/CLI/COAP/coap_switch/ci.json similarity index 100% rename from libraries/OpenThread/examples/COAP/coap_switch/ci.json rename to libraries/OpenThread/examples/CLI/COAP/coap_switch/ci.json diff --git a/libraries/OpenThread/examples/COAP/coap_switch/coap_switch.ino b/libraries/OpenThread/examples/CLI/COAP/coap_switch/coap_switch.ino similarity index 95% rename from libraries/OpenThread/examples/COAP/coap_switch/coap_switch.ino rename to libraries/OpenThread/examples/CLI/COAP/coap_switch/coap_switch.ino index aac5db0bc82..bacac47ddeb 100644 --- a/libraries/OpenThread/examples/COAP/coap_switch/coap_switch.ino +++ b/libraries/OpenThread/examples/CLI/COAP/coap_switch/coap_switch.ino @@ -69,18 +69,18 @@ bool otDeviceSetup( Serial.println("OpenThread started.\r\nWaiting for activating correct Device Role."); // wait for the expected Device Role to start uint8_t tries = 24; // 24 x 2.5 sec = 1 min - while (tries && otGetDeviceRole() != expectedRole1 && otGetDeviceRole() != expectedRole2) { + while (tries && OThread.otGetDeviceRole() != expectedRole1 && OThread.otGetDeviceRole() != expectedRole2) { Serial.print("."); delay(2500); tries--; } Serial.println(); if (!tries) { - log_e("Sorry, Device Role failed by timeout! Current Role: %s.", otGetStringDeviceRole()); + log_e("Sorry, Device Role failed by timeout! Current Role: %s.", OThread.otGetStringDeviceRole()); rgbLedWrite(RGB_BUILTIN, 255, 0, 0); // RED ... failed! return false; } - Serial.printf("Device is %s.\r\n", otGetStringDeviceRole()); + Serial.printf("Device is %s.\r\n", OThread.otGetStringDeviceRole()); for (i = 0; i < nCmds2; i++) { if (!otExecCommand(otCoapCmds[i * 2], otCoapCmds[i * 2 + 1])) { break; @@ -176,7 +176,8 @@ void setup() { Serial.begin(115200); // LED starts RED, indicating not connected to Thread network. rgbLedWrite(RGB_BUILTIN, 64, 0, 0); - OThreadCLI.begin(false); // No AutoStart is necessary + OThread.begin(false); // No AutoStart is necessary + OThreadCLI.begin(); OThreadCLI.setTimeout(250); // waits 250ms for the OpenThread CLI response setupNode(); // LED goes and keeps Blue when all is ready and Red when failed. diff --git a/libraries/OpenThread/examples/SimpleCLI/SimpleCLI.ino b/libraries/OpenThread/examples/CLI/SimpleCLI/SimpleCLI.ino similarity index 89% rename from libraries/OpenThread/examples/SimpleCLI/SimpleCLI.ino rename to libraries/OpenThread/examples/CLI/SimpleCLI/SimpleCLI.ino index feef800c0fa..dec3aaeb218 100644 --- a/libraries/OpenThread/examples/SimpleCLI/SimpleCLI.ino +++ b/libraries/OpenThread/examples/CLI/SimpleCLI/SimpleCLI.ino @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. @@ -26,7 +26,8 @@ void setup() { Serial.begin(115200); - OThreadCLI.begin(false); // No AutoStart - fresh start + OThread.begin(false); // No AutoStart - fresh start + OThreadCLI.begin(); Serial.println("OpenThread CLI started - type 'help' for a list of commands."); OThreadCLI.startConsole(Serial); } diff --git a/libraries/OpenThread/examples/SimpleCLI/ci.json b/libraries/OpenThread/examples/CLI/SimpleCLI/ci.json similarity index 100% rename from libraries/OpenThread/examples/SimpleCLI/ci.json rename to libraries/OpenThread/examples/CLI/SimpleCLI/ci.json diff --git a/libraries/OpenThread/examples/SimpleNode/SimpleNode.ino b/libraries/OpenThread/examples/CLI/SimpleNode/SimpleNode.ino similarity index 82% rename from libraries/OpenThread/examples/SimpleNode/SimpleNode.ino rename to libraries/OpenThread/examples/CLI/SimpleNode/SimpleNode.ino index 95bf7a2401a..d17b692cc74 100644 --- a/libraries/OpenThread/examples/SimpleNode/SimpleNode.ino +++ b/libraries/OpenThread/examples/CLI/SimpleNode/SimpleNode.ino @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. @@ -35,12 +35,13 @@ void setup() { Serial.begin(115200); - OThreadCLI.begin(); // AutoStart using Thread default settings - otPrintNetworkInformation(Serial); // Print Current Thread Network Information + OThread.begin(); // AutoStart using Thread default settings + OThreadCLI.begin(); + OThread.otPrintNetworkInformation(Serial); // Print Current Thread Network Information } void loop() { Serial.print("Thread Node State: "); - Serial.println(otGetStringDeviceRole()); + Serial.println(OThread.otGetStringDeviceRole()); delay(5000); } diff --git a/libraries/OpenThread/examples/SimpleNode/ci.json b/libraries/OpenThread/examples/CLI/SimpleNode/ci.json similarity index 100% rename from libraries/OpenThread/examples/SimpleNode/ci.json rename to libraries/OpenThread/examples/CLI/SimpleNode/ci.json diff --git a/libraries/OpenThread/examples/SimpleThreadNetwork/ExtendedRouterNode/ExtendedRouterNode.ino b/libraries/OpenThread/examples/CLI/SimpleThreadNetwork/ExtendedRouterNode/ExtendedRouterNode.ino similarity index 89% rename from libraries/OpenThread/examples/SimpleThreadNetwork/ExtendedRouterNode/ExtendedRouterNode.ino rename to libraries/OpenThread/examples/CLI/SimpleThreadNetwork/ExtendedRouterNode/ExtendedRouterNode.ino index 4fc8a921584..40f046aeab5 100644 --- a/libraries/OpenThread/examples/SimpleThreadNetwork/ExtendedRouterNode/ExtendedRouterNode.ino +++ b/libraries/OpenThread/examples/CLI/SimpleThreadNetwork/ExtendedRouterNode/ExtendedRouterNode.ino @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. @@ -22,7 +22,8 @@ bool otStatus = true; void setup() { Serial.begin(115200); - OThreadCLI.begin(false); // No AutoStart - fresh start + OThread.begin(false); // No AutoStart - fresh start + OThreadCLI.begin(); Serial.println("Setting up OpenThread Node as Router/Child"); Serial.println("Make sure the Leader Node is already running"); @@ -39,7 +40,7 @@ void setup() { } // wait for the node to enter in the router state uint32_t timeout = millis() + 90000; // waits 90 seconds to - while (otGetDeviceRole() != OT_ROLE_CHILD && otGetDeviceRole() != OT_ROLE_ROUTER) { + while (OThread.otGetDeviceRole() != OT_ROLE_CHILD && OThread.otGetDeviceRole() != OT_ROLE_ROUTER) { Serial.print("."); if (millis() > timeout) { Serial.println("\r\n\t===> Timeout! Failed."); @@ -70,7 +71,7 @@ void loop() { if (otStatus) { Serial.println("Thread NetworkInformation: "); Serial.println("---------------------------"); - otPrintNetworkInformation(Serial); + OThread.otPrintNetworkInformation(Serial); Serial.println("---------------------------"); } else { Serial.println("Some OpenThread operation has failed..."); diff --git a/libraries/OpenThread/examples/SimpleThreadNetwork/ExtendedRouterNode/ci.json b/libraries/OpenThread/examples/CLI/SimpleThreadNetwork/ExtendedRouterNode/ci.json similarity index 100% rename from libraries/OpenThread/examples/SimpleThreadNetwork/ExtendedRouterNode/ci.json rename to libraries/OpenThread/examples/CLI/SimpleThreadNetwork/ExtendedRouterNode/ci.json diff --git a/libraries/OpenThread/examples/SimpleThreadNetwork/LeaderNode/LeaderNode.ino b/libraries/OpenThread/examples/CLI/SimpleThreadNetwork/LeaderNode/LeaderNode.ino similarity index 93% rename from libraries/OpenThread/examples/SimpleThreadNetwork/LeaderNode/LeaderNode.ino rename to libraries/OpenThread/examples/CLI/SimpleThreadNetwork/LeaderNode/LeaderNode.ino index 7b709717692..a945a5c8f77 100644 --- a/libraries/OpenThread/examples/SimpleThreadNetwork/LeaderNode/LeaderNode.ino +++ b/libraries/OpenThread/examples/CLI/SimpleThreadNetwork/LeaderNode/LeaderNode.ino @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. @@ -35,7 +35,8 @@ otInstance *aInstance = NULL; void setup() { Serial.begin(115200); - OThreadCLI.begin(false); // No AutoStart - fresh start + OThread.begin(false); // No AutoStart - fresh start + OThreadCLI.begin(); Serial.println(); Serial.println("Setting up OpenThread Node as Leader"); aInstance = esp_openthread_get_instance(); @@ -51,11 +52,11 @@ void setup() { void loop() { Serial.println("============================================="); Serial.print("Thread Node State: "); - Serial.println(otGetStringDeviceRole()); + Serial.println(OThread.otGetStringDeviceRole()); // Native OpenThread API calls: // wait until the node become Child or Router - if (otGetDeviceRole() == OT_ROLE_LEADER) { + if (OThread.otGetDeviceRole() == OT_ROLE_LEADER) { // Network Name const char *networkName = otThreadGetNetworkName(aInstance); Serial.printf("Network Name: %s\r\n", networkName); diff --git a/libraries/OpenThread/examples/SimpleThreadNetwork/LeaderNode/ci.json b/libraries/OpenThread/examples/CLI/SimpleThreadNetwork/LeaderNode/ci.json similarity index 100% rename from libraries/OpenThread/examples/SimpleThreadNetwork/LeaderNode/ci.json rename to libraries/OpenThread/examples/CLI/SimpleThreadNetwork/LeaderNode/ci.json diff --git a/libraries/OpenThread/examples/SimpleThreadNetwork/RouterNode/RouterNode.ino b/libraries/OpenThread/examples/CLI/SimpleThreadNetwork/RouterNode/RouterNode.ino similarity index 92% rename from libraries/OpenThread/examples/SimpleThreadNetwork/RouterNode/RouterNode.ino rename to libraries/OpenThread/examples/CLI/SimpleThreadNetwork/RouterNode/RouterNode.ino index 45475fa0c6a..f802bd7ef01 100644 --- a/libraries/OpenThread/examples/SimpleThreadNetwork/RouterNode/RouterNode.ino +++ b/libraries/OpenThread/examples/CLI/SimpleThreadNetwork/RouterNode/RouterNode.ino @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. @@ -34,7 +34,8 @@ otInstance *aInstance = NULL; void setup() { Serial.begin(115200); - OThreadCLI.begin(false); // No AutoStart - fresh start + OThread.begin(false); // No AutoStart - fresh start + OThreadCLI.begin(); Serial.println(); Serial.println("Setting up OpenThread Node as Router/Child"); Serial.println("Make sure the Leader Node is already running"); @@ -51,11 +52,11 @@ void setup() { void loop() { Serial.println("============================================="); Serial.print("Thread Node State: "); - Serial.println(otGetStringDeviceRole()); + Serial.println(OThread.otGetStringDeviceRole()); // Native OpenThread API calls: // wait until the node become Child or Router - if (otGetDeviceRole() == OT_ROLE_CHILD || otGetDeviceRole() == OT_ROLE_ROUTER) { + if (OThread.otGetDeviceRole() == OT_ROLE_CHILD || OThread.otGetDeviceRole() == OT_ROLE_ROUTER) { // Network Name const char *networkName = otThreadGetNetworkName(aInstance); Serial.printf("Network Name: %s\r\n", networkName); diff --git a/libraries/OpenThread/examples/SimpleThreadNetwork/RouterNode/ci.json b/libraries/OpenThread/examples/CLI/SimpleThreadNetwork/RouterNode/ci.json similarity index 100% rename from libraries/OpenThread/examples/SimpleThreadNetwork/RouterNode/ci.json rename to libraries/OpenThread/examples/CLI/SimpleThreadNetwork/RouterNode/ci.json diff --git a/libraries/OpenThread/examples/ThreadScan/ThreadScan.ino b/libraries/OpenThread/examples/CLI/ThreadScan/ThreadScan.ino similarity index 89% rename from libraries/OpenThread/examples/ThreadScan/ThreadScan.ino rename to libraries/OpenThread/examples/CLI/ThreadScan/ThreadScan.ino index 9d0074bb180..87c29339fb7 100644 --- a/libraries/OpenThread/examples/ThreadScan/ThreadScan.ino +++ b/libraries/OpenThread/examples/CLI/ThreadScan/ThreadScan.ino @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. @@ -27,7 +27,8 @@ void setup() { Serial.begin(115200); - OThreadCLI.begin(true); // For scanning, AutoStart must be active, any setup + OThread.begin(true); // For scanning, AutoStart must be active, any setup + OThreadCLI.begin(); OThreadCLI.setTimeout(100); // Set a timeout for the CLI response Serial.println(); Serial.println("This sketch will continuously scan the Thread Local Network and all devices IEEE 802.15.4 compatible"); @@ -41,7 +42,7 @@ void loop() { Serial.println("Scan Failed..."); } delay(5000); - if (otGetDeviceRole() < OT_ROLE_CHILD) { + if (OThread.otGetDeviceRole() < OT_ROLE_CHILD) { Serial.println(); Serial.println("This device has not started Thread yet, bypassing Discovery Scan"); return; diff --git a/libraries/OpenThread/examples/ThreadScan/ci.json b/libraries/OpenThread/examples/CLI/ThreadScan/ci.json similarity index 100% rename from libraries/OpenThread/examples/ThreadScan/ci.json rename to libraries/OpenThread/examples/CLI/ThreadScan/ci.json diff --git a/libraries/OpenThread/examples/onReceive/ci.json b/libraries/OpenThread/examples/CLI/onReceive/ci.json similarity index 100% rename from libraries/OpenThread/examples/onReceive/ci.json rename to libraries/OpenThread/examples/CLI/onReceive/ci.json diff --git a/libraries/OpenThread/examples/onReceive/onReceive.ino b/libraries/OpenThread/examples/CLI/onReceive/onReceive.ino similarity index 96% rename from libraries/OpenThread/examples/onReceive/onReceive.ino rename to libraries/OpenThread/examples/CLI/onReceive/onReceive.ino index b37c2fc7931..f53cc33f5ec 100644 --- a/libraries/OpenThread/examples/onReceive/onReceive.ino +++ b/libraries/OpenThread/examples/CLI/onReceive/onReceive.ino @@ -39,7 +39,8 @@ void otReceivedLine() { void setup() { Serial.begin(115200); - OThreadCLI.begin(); // AutoStart + OThread.begin(); // AutoStart + OThreadCLI.begin(); OThreadCLI.onReceive(otReceivedLine); } diff --git a/libraries/OpenThread/examples/Native/SimpleThreadNetwork/LeaderNode/LeaderNode.ino b/libraries/OpenThread/examples/Native/SimpleThreadNetwork/LeaderNode/LeaderNode.ino new file mode 100644 index 00000000000..dfea9776838 --- /dev/null +++ b/libraries/OpenThread/examples/Native/SimpleThreadNetwork/LeaderNode/LeaderNode.ino @@ -0,0 +1,34 @@ +#include "OThread.h" + +OpenThread threadLeaderNode; +DataSet dataset; + +void setup() { + Serial.begin(115200); + + // Start OpenThread Stack - false for not using NVS dataset information + threadLeaderNode.begin(false); + + // Create a new Thread Network Dataset for a Leader Node + dataset.initNew(); + // Configure the dataset + dataset.setNetworkName("ESP_OpenThread"); + uint8_t extPanId[OT_EXT_PAN_ID_SIZE] = {0xDE, 0xAD, 0x00, 0xBE, 0xEF, 0x00, 0xCA, 0xFE}; + dataset.setExtendedPanId(extPanId); + uint8_t networkKey[OT_NETWORK_KEY_SIZE] = {0x00, 0x11, 0x22, 0x33, 0x44, 0x55, 0x66, 0x77, 0x88, 0x99, 0xaa, 0xbb, 0xcc, 0xdd, 0xee, 0xff}; + dataset.setNetworkKey(networkKey); + dataset.setChannel(15); + dataset.setPanId(0x1234); + + // Apply the dataset and start the network + threadLeaderNode.commitDataSet(dataset); + threadLeaderNode.networkInterfaceUp(); + threadLeaderNode.start(); +} + +void loop() { + // Print network information every 5 seconds + Serial.println("=============================================="); + threadLeaderNode.otPrintNetworkInformation(Serial); + delay(5000); +} diff --git a/libraries/OpenThread/examples/Native/SimpleThreadNetwork/LeaderNode/ci.json b/libraries/OpenThread/examples/Native/SimpleThreadNetwork/LeaderNode/ci.json new file mode 100644 index 00000000000..2ee6af3490e --- /dev/null +++ b/libraries/OpenThread/examples/Native/SimpleThreadNetwork/LeaderNode/ci.json @@ -0,0 +1,6 @@ +{ + "requires": [ + "CONFIG_OPENTHREAD_ENABLED=y", + "CONFIG_SOC_IEEE802154_SUPPORTED=y" + ] +} diff --git a/libraries/OpenThread/examples/Native/SimpleThreadNetwork/RouterNode/RouterNode.ino b/libraries/OpenThread/examples/Native/SimpleThreadNetwork/RouterNode/RouterNode.ino new file mode 100644 index 00000000000..5ffa535ad51 --- /dev/null +++ b/libraries/OpenThread/examples/Native/SimpleThreadNetwork/RouterNode/RouterNode.ino @@ -0,0 +1,29 @@ +#include "OThread.h" + +OpenThread threadChildNode; +DataSet dataset; + +void setup() { + Serial.begin(115200); + + // Start OpenThread Stack - false for not using NVS dataset information + threadChildNode.begin(false); + + // clear dataset + dataset.clear(); + // Configure the dataset with the same Network Key of the Leader Node + uint8_t networkKey[OT_NETWORK_KEY_SIZE] = {0x00, 0x11, 0x22, 0x33, 0x44, 0x55, 0x66, 0x77, 0x88, 0x99, 0xaa, 0xbb, 0xcc, 0xdd, 0xee, 0xff}; + dataset.setNetworkKey(networkKey); + + // Apply the dataset and start the network + threadChildNode.commitDataSet(dataset); + threadChildNode.networkInterfaceUp(); + threadChildNode.start(); +} + +void loop() { + // Print network information every 5 seconds + Serial.println("=============================================="); + threadChildNode.otPrintNetworkInformation(Serial); + delay(5000); +} diff --git a/libraries/OpenThread/examples/Native/SimpleThreadNetwork/RouterNode/ci.json b/libraries/OpenThread/examples/Native/SimpleThreadNetwork/RouterNode/ci.json new file mode 100644 index 00000000000..2ee6af3490e --- /dev/null +++ b/libraries/OpenThread/examples/Native/SimpleThreadNetwork/RouterNode/ci.json @@ -0,0 +1,6 @@ +{ + "requires": [ + "CONFIG_OPENTHREAD_ENABLED=y", + "CONFIG_SOC_IEEE802154_SUPPORTED=y" + ] +} diff --git a/libraries/OpenThread/keywords.txt b/libraries/OpenThread/keywords.txt index d7193de188d..b62c2c23ddc 100644 --- a/libraries/OpenThread/keywords.txt +++ b/libraries/OpenThread/keywords.txt @@ -7,7 +7,10 @@ ####################################### OThreadCLI KEYWORD1 +OThread KEYWORD1 OpenThreadCLI KEYWORD1 +OpenThread KEYWORD1 +DataSet KEYWORD1 ot_cmd_return_t KEYWORD1 ot_device_role_t KEYWORD1 @@ -35,6 +38,27 @@ otGetRespCmd KEYWORD2 otExecCommand KEYWORD2 otPrintRespCLI KEYWORD2 otPrintNetworkInformation KEYWORD2 +clear KEYWORD2 +initNew KEYWORD2 +getDataset KEYWORD2 +setNetworkName KEYWORD2 +getNetworkName KEYWORD2 +setExtendedPanId KEYWORD2 +getExtendedPanId KEYWORD2 +setNetworkKey KEYWORD2 +getNetworkKey KEYWORD2 +setChannel KEYWORD2 +getChannel KEYWORD2 +setPanId KEYWORD2 +getPanId KEYWORD2 +apply KEYWORD2 +otStarted KEYWORD2 +otCLIStarted KEYWORD2 +start KEYWORD2 +stop KEYWORD2 +networkInterfaceUp KEYWORD2 +networkInterfaceDown KEYWORD2 +commitDataSet KEYWORD2 ####################################### # Constants (LITERAL1) @@ -45,3 +69,5 @@ OT_ROLE_DETACHED LITERAL1 OT_ROLE_CHILD LITERAL1 OT_ROLE_ROUTER LITERAL1 OT_ROLE_LEADER LITERAL1 +OT_EXT_PAN_ID_SIZE LITERAL1 +OT_NETWORK_KEY_SIZE LITERAL1 diff --git a/libraries/OpenThread/src/OThread.cpp b/libraries/OpenThread/src/OThread.cpp new file mode 100644 index 00000000000..43d8ce02918 --- /dev/null +++ b/libraries/OpenThread/src/OThread.cpp @@ -0,0 +1,364 @@ +#include "OThread.h" +#if SOC_IEEE802154_SUPPORTED +#if CONFIG_OPENTHREAD_ENABLED + +#include "esp_err.h" +#include "esp_event.h" +#include "esp_netif.h" +#include "esp_netif_types.h" +#include "esp_vfs_eventfd.h" + +#include "freertos/FreeRTOS.h" +#include "freertos/task.h" +#include "freertos/semphr.h" +#include "freertos/queue.h" + +#include "esp_netif_net_stack.h" +#include "esp_openthread_netif_glue.h" +#include "lwip/netif.h" + +static esp_openthread_platform_config_t ot_native_config; +static esp_netif_t *openthread_netif = NULL; + +const char *otRoleString[] = { + "Disabled", ///< The Thread stack is disabled. + "Detached", ///< Not currently participating in a Thread network/partition. + "Child", ///< The Thread Child role. + "Router", ///< The Thread Router role. + "Leader", ///< The Thread Leader role. + "Unknown", ///< Unknown role, not initialized or not started. +}; + +static TaskHandle_t s_ot_task = NULL; +#if CONFIG_LWIP_HOOK_IP6_INPUT_CUSTOM +static struct netif *ot_lwip_netif = NULL; +#endif + +#if CONFIG_LWIP_HOOK_IP6_INPUT_CUSTOM +extern "C" int lwip_hook_ip6_input(struct pbuf *p, struct netif *inp) { + if (ot_lwip_netif && ot_lwip_netif == inp) { + return 0; + } + if (ip6_addr_isany_val(inp->ip6_addr[0].u_addr.ip6)) { + // We don't have an LL address -> eat this packet here, so it won't get accepted on input netif + pbuf_free(p); + return 1; + } + return 0; +} +#endif + +static void ot_task_worker(void *aContext) { + esp_vfs_eventfd_config_t eventfd_config = { + .max_fds = 3, + }; + bool err = false; + if (ESP_OK != esp_event_loop_create_default()) { + log_e("Failed to create OpentThread event loop"); + err = true; + } + if (!err && ESP_OK != esp_netif_init()) { + log_e("Failed to initialize OpentThread netif"); + err = true; + } + if (!err && ESP_OK != esp_vfs_eventfd_register(&eventfd_config)) { + log_e("Failed to register OpentThread eventfd"); + err = true; + } + + // Initialize the OpenThread stack + if (!err && ESP_OK != esp_openthread_init(&ot_native_config)) { + log_e("Failed to initialize OpenThread stack"); + err = true; + } + if (!err) { + // Initialize the esp_netif bindings + esp_netif_config_t cfg = ESP_NETIF_DEFAULT_OPENTHREAD(); + openthread_netif = esp_netif_new(&cfg); +#if CONFIG_LWIP_HOOK_IP6_INPUT_CUSTOM + // Get LwIP Netif + if (openthread_netif != NULL) { + ot_lwip_netif = (struct netif *)esp_netif_get_netif_impl(openthread_netif); + if (ot_lwip_netif == NULL) { + log_e("Failed to get OpenThread LwIP netif"); + } + } +#endif + } + if (!err && openthread_netif == NULL) { + log_e("Failed to create OpenThread esp_netif"); + err = true; + } + if (!err && ESP_OK != esp_netif_attach(openthread_netif, esp_openthread_netif_glue_init(&ot_native_config))) { + log_e("Failed to attach OpenThread esp_netif"); + err = true; + } + if (!err && ESP_OK != esp_netif_set_default_netif(openthread_netif)) { + log_e("Failed to set default OpenThread esp_netif"); + err = true; + } + if (!err) { + // only returns in case there is an OpenThread Stack failure... + esp_openthread_launch_mainloop(); + } + // Clean up + esp_openthread_netif_glue_deinit(); + esp_netif_destroy(openthread_netif); + esp_vfs_eventfd_unregister(); + vTaskDelete(NULL); +} + +// DataSet Implementation +DataSet::DataSet() { + memset(&mDataset, 0, sizeof(mDataset)); +} + +void DataSet::clear() { + memset(&mDataset, 0, sizeof(mDataset)); +} + +void DataSet::initNew() { + otInstance *mInstance = esp_openthread_get_instance(); + if (!mInstance) { + log_e("OpenThread not started. Please begin() it before initializing a new dataset."); + return; + } + clear(); + otDatasetCreateNewNetwork(mInstance, &mDataset); +} + +const otOperationalDataset &DataSet::getDataset() const { + return mDataset; +} + +void DataSet::setNetworkName(const char *name) { + strncpy(mDataset.mNetworkName.m8, name, sizeof(mDataset.mNetworkName.m8)); + mDataset.mComponents.mIsNetworkNamePresent = true; +} + +void DataSet::setExtendedPanId(const uint8_t *extPanId) { + memcpy(mDataset.mExtendedPanId.m8, extPanId, OT_EXT_PAN_ID_SIZE); + mDataset.mComponents.mIsExtendedPanIdPresent = true; +} + +void DataSet::setNetworkKey(const uint8_t *key) { + memcpy(mDataset.mNetworkKey.m8, key, OT_NETWORK_KEY_SIZE); + mDataset.mComponents.mIsNetworkKeyPresent = true; +} + +void DataSet::setChannel(uint8_t channel) { + mDataset.mChannel = channel; + mDataset.mComponents.mIsChannelPresent = true; +} + +void DataSet::setPanId(uint16_t panId) { + mDataset.mPanId = panId; + mDataset.mComponents.mIsPanIdPresent = true; +} + +const char *DataSet::getNetworkName() const { + return mDataset.mNetworkName.m8; +} + +const uint8_t *DataSet::getExtendedPanId() const { + return mDataset.mExtendedPanId.m8; +} + +const uint8_t *DataSet::getNetworkKey() const { + return mDataset.mNetworkKey.m8; +} + +uint8_t DataSet::getChannel() const { + return mDataset.mChannel; +} + +uint16_t DataSet::getPanId() const { + return mDataset.mPanId; +} + +void DataSet::apply(otInstance *instance) { + otDatasetSetActive(instance, &mDataset); +} + +// OpenThread Implementation +bool OpenThread::otStarted = false; + +otInstance *OpenThread::mInstance = nullptr; +OpenThread::OpenThread() {} + +OpenThread::~OpenThread() { + end(); +} + +OpenThread::operator bool() const { + return otStarted; +} + +void OpenThread::begin(bool OThreadAutoStart) { + if (otStarted) { + log_w("OpenThread already started"); + return; + } + + memset(&ot_native_config, 0, sizeof(esp_openthread_platform_config_t)); + ot_native_config.radio_config.radio_mode = RADIO_MODE_NATIVE; + ot_native_config.host_config.host_connection_mode = HOST_CONNECTION_MODE_NONE; + ot_native_config.port_config.storage_partition_name = "nvs"; + ot_native_config.port_config.netif_queue_size = 10; + ot_native_config.port_config.task_queue_size = 10; + + // Initialize OpenThread stack + xTaskCreate(ot_task_worker, "ot_main_loop", 10240, NULL, 20, &s_ot_task); + if (s_ot_task == NULL) { + log_e("Error: Failed to create OpenThread task"); + return; + } + log_d("OpenThread task created successfully"); + // get the OpenThread instance that will be used for all operations + mInstance = esp_openthread_get_instance(); + if (!mInstance) { + log_e("Error: Failed to initialize OpenThread instance"); + end(); + return; + } + // starts Thread with default dataset from NVS or from IDF default settings + if (OThreadAutoStart) { + otOperationalDatasetTlvs dataset; + otError error = otDatasetGetActiveTlvs(esp_openthread_get_instance(), &dataset); + // error = OT_ERROR_FAILED; // teste para forçar NULL dataset + if (error != OT_ERROR_NONE) { + log_i("Failed to get active NVS dataset from OpenThread"); + } else { + log_i("Got active NVS dataset from OpenThread"); + } + esp_err_t err = esp_openthread_auto_start((error == OT_ERROR_NONE) ? &dataset : NULL); + if (err != ESP_OK) { + log_i("Failed to AUTO start OpenThread"); + } else { + log_i("AUTO start OpenThread done"); + } + } + otStarted = true; +} + +void OpenThread::end() { + if (s_ot_task != NULL) { + vTaskDelete(s_ot_task); + s_ot_task = NULL; + // Clean up + esp_openthread_deinit(); + esp_openthread_netif_glue_deinit(); +#if CONFIG_LWIP_HOOK_IP6_INPUT_CUSTOM + ot_lwip_netif = NULL; +#endif + esp_netif_destroy(openthread_netif); + esp_vfs_eventfd_unregister(); + } + otStarted = false; +} + +void OpenThread::start() { + if (!mInstance) { + log_w("Error: OpenThread instance not initialized"); + return; + } + otThreadSetEnabled(mInstance, true); + log_d("Thread network started"); +} + +void OpenThread::stop() { + if (!mInstance) { + log_w("Error: OpenThread instance not initialized"); + return; + } + otThreadSetEnabled(mInstance, false); + log_d("Thread network stopped"); +} + +void OpenThread::networkInterfaceUp() { + if (!mInstance) { + log_w("Error: OpenThread instance not initialized"); + return; + } + // Enable the Thread interface (equivalent to CLI Command "ifconfig up") + otError error = otIp6SetEnabled(mInstance, true); + if (error != OT_ERROR_NONE) { + log_e("Error: Failed to enable Thread interface (error code: %d)\n", error); + } + log_d("OpenThread Network Interface is up"); +} + +void OpenThread::networkInterfaceDown() { + if (!mInstance) { + log_w("Error: OpenThread instance not initialized"); + return; + } + // Disable the Thread interface (equivalent to CLI Command "ifconfig down") + otError error = otIp6SetEnabled(mInstance, false); + if (error != OT_ERROR_NONE) { + log_e("Error: Failed to disable Thread interface (error code: %d)\n", error); + } + log_d("OpenThread Network Interface is down"); +} + +void OpenThread::commitDataSet(const DataSet &dataset) { + if (!mInstance) { + log_w("Error: OpenThread instance not initialized"); + return; + } + // Commit the dataset as the active dataset + otError error = otDatasetSetActive(mInstance, &(dataset.getDataset())); + if (error != OT_ERROR_NONE) { + log_e("Error: Failed to commit dataset (error code: %d)\n", error); + return; + } + log_d("Dataset committed successfully"); +} + +ot_device_role_t OpenThread::otGetDeviceRole() { + if (!mInstance) { + log_w("Error: OpenThread instance not initialized"); + return OT_ROLE_DISABLED; + } + return (ot_device_role_t)otThreadGetDeviceRole(mInstance); +} + +const char *OpenThread::otGetStringDeviceRole() { + return otRoleString[otGetDeviceRole()]; +} + +void OpenThread::otPrintNetworkInformation(Stream &output) { + if (!mInstance) { + log_w("Error: OpenThread instance not initialized"); + return; + } + + output.printf("Role: %s", otGetStringDeviceRole()); + output.println(); + output.printf("Network Name: %s", otThreadGetNetworkName(mInstance)); + output.println(); + output.printf("Channel: %d", otLinkGetChannel(mInstance)); + output.println(); + output.printf("PAN ID: 0x%04x", otLinkGetPanId(mInstance)); + output.println(); + + const otExtendedPanId *extPanId = otThreadGetExtendedPanId(mInstance); + output.print("Extended PAN ID: "); + for (int i = 0; i < OT_EXT_PAN_ID_SIZE; i++) { + output.printf("%02x", extPanId->m8[i]); + } + output.println(); + + otNetworkKey networkKey; + otThreadGetNetworkKey(mInstance, &networkKey); + output.print("Network Key: "); + for (int i = 0; i < OT_NETWORK_KEY_SIZE; i++) { + output.printf("%02x", networkKey.m8[i]); + } + output.println(); +} + +OpenThread OThread; + +#endif /* CONFIG_OPENTHREAD_ENABLED */ +#endif /* SOC_IEEE802154_SUPPORTED */ diff --git a/libraries/OpenThread/src/OThread.h b/libraries/OpenThread/src/OThread.h new file mode 100644 index 00000000000..359d581bb9d --- /dev/null +++ b/libraries/OpenThread/src/OThread.h @@ -0,0 +1,107 @@ +// 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. + +#pragma once +#include "soc/soc_caps.h" +#include "sdkconfig.h" +#if SOC_IEEE802154_SUPPORTED +#if CONFIG_OPENTHREAD_ENABLED + +#include +#include +#include +#include +#include +#include +#include + +typedef enum { + OT_ROLE_DISABLED = 0, ///< The Thread stack is disabled. + OT_ROLE_DETACHED = 1, ///< Not currently participating in a Thread network/partition. + OT_ROLE_CHILD = 2, ///< The Thread Child role. + OT_ROLE_ROUTER = 3, ///< The Thread Router role. + OT_ROLE_LEADER = 4, ///< The Thread Leader role. +} ot_device_role_t; +extern const char *otRoleString[]; + +class DataSet { +public: + DataSet(); + void clear(); + void initNew(); + const otOperationalDataset &getDataset() const; + + // Setters + void setNetworkName(const char *name); + void setExtendedPanId(const uint8_t *extPanId); + void setNetworkKey(const uint8_t *key); + void setChannel(uint8_t channel); + void setPanId(uint16_t panId); + + // Getters + const char *getNetworkName() const; + const uint8_t *getExtendedPanId() const; + const uint8_t *getNetworkKey() const; + uint8_t getChannel() const; + uint16_t getPanId() const; + + // Apply the dataset to the OpenThread instance + void apply(otInstance *instance); + +private: + otOperationalDataset mDataset; +}; + +class OpenThread { +public: + static bool otStarted; + static ot_device_role_t otGetDeviceRole(); + static const char *otGetStringDeviceRole(); + static void otPrintNetworkInformation(Stream &output); + + OpenThread(); + ~OpenThread(); + // returns true if OpenThread Stack is running + operator bool() const; + + // Initialize OpenThread + static void begin(bool OThreadAutoStart = true); + + // Initialize OpenThread + static void end(); + + // Start the Thread network + void start(); + + // Stop the Thread network + void stop(); + + // Start Thread Network Interface + void networkInterfaceUp(); + + // Stop Thread Network Interface + void networkInterfaceDown(); + + // Set the dataset + void commitDataSet(const DataSet &dataset); + +private: + static otInstance *mInstance; + DataSet mCurrentDataSet; +}; + +extern OpenThread OThread; + +#endif /* CONFIG_OPENTHREAD_ENABLED */ +#endif /* SOC_IEEE802154_SUPPORTED */ diff --git a/libraries/OpenThread/src/OThreadCLI.cpp b/libraries/OpenThread/src/OThreadCLI.cpp index 85ba03563e8..2f0f97a0539 100644 --- a/libraries/OpenThread/src/OThreadCLI.cpp +++ b/libraries/OpenThread/src/OThreadCLI.cpp @@ -33,18 +33,12 @@ #include "esp_netif_net_stack.h" #include "lwip/netif.h" +bool OpenThreadCLI::otCLIStarted = false; static TaskHandle_t s_cli_task = NULL; static TaskHandle_t s_console_cli_task = NULL; static QueueHandle_t rx_queue = NULL; static QueueHandle_t tx_queue = NULL; -static esp_openthread_platform_config_t ot_native_config; -static TaskHandle_t s_ot_task = NULL; -static esp_netif_t *openthread_netif = NULL; -#if CONFIG_LWIP_HOOK_IP6_INPUT_CUSTOM -static struct netif *ot_lwip_netif = NULL; -#endif - #define OT_CLI_MAX_LINE_LENGTH 512 typedef struct { @@ -53,21 +47,7 @@ typedef struct { String prompt; OnReceiveCb_t responseCallBack; } ot_cli_console_t; -static ot_cli_console_t otConsole = {NULL, false, (const char *)NULL, NULL}; - -#if CONFIG_LWIP_HOOK_IP6_INPUT_CUSTOM -extern "C" int lwip_hook_ip6_input(struct pbuf *p, struct netif *inp) { - if (ot_lwip_netif && ot_lwip_netif == inp) { - return 0; - } - if (ip6_addr_isany_val(inp->ip6_addr[0].u_addr.ip6)) { - // We don't have an LL address -> eat this packet here, so it won't get accepted on input netif - pbuf_free(p); - return 1; - } - return 0; -} -#endif +static ot_cli_console_t otConsole = {nullptr, false, (const char *)nullptr, nullptr}; // process the CLI commands sent to the OpenThread stack static void ot_cli_loop(void *context) { @@ -131,7 +111,7 @@ static int ot_cli_output_callback(void *context, const char *format, va_list arg } // if there is a user callback function in place, it shall have the priority // to process/consume the Stream data received from OpenThread CLI, which is available in its RX Buffer - if (otConsole.responseCallBack != NULL) { + if (otConsole.responseCallBack != nullptr) { otConsole.responseCallBack(); } } @@ -205,7 +185,7 @@ void OpenThreadCLI::setEchoBack(bool echoback) { } void OpenThreadCLI::setPrompt(char *prompt) { - otConsole.prompt = prompt; // NULL will make the prompt not visible + otConsole.prompt = prompt; // nullptr can make the prompt not visible } void OpenThreadCLI::setStream(Stream &otStream) { @@ -213,12 +193,12 @@ void OpenThreadCLI::setStream(Stream &otStream) { } void OpenThreadCLI::onReceive(OnReceiveCb_t func) { - otConsole.responseCallBack = func; // NULL will set it off + otConsole.responseCallBack = func; // nullptr will set it off } // Stream object shall be already started and configured before calling this function void OpenThreadCLI::startConsole(Stream &otStream, bool echoback, const char *prompt) { - if (!otStarted) { + if (!otCLIStarted) { log_e("OpenThread CLI has not started. Please begin() it before starting the console."); return; } @@ -226,7 +206,7 @@ void OpenThreadCLI::startConsole(Stream &otStream, bool echoback, const char *pr if (s_console_cli_task == NULL) { otConsole.cliStream = &otStream; otConsole.echoback = echoback; - otConsole.prompt = prompt; // NULL will invalidate the String + otConsole.prompt = prompt; // nullptr will invalidate the String // it will run in the same priority (1) as the Arduino setup()/loop() task xTaskCreate(ot_cli_console_worker, "ot_cli_console", 4096, &otConsole, 1, &s_console_cli_task); } else { @@ -242,12 +222,6 @@ void OpenThreadCLI::stopConsole() { } OpenThreadCLI::OpenThreadCLI() { - memset(&ot_native_config, 0, sizeof(esp_openthread_platform_config_t)); - ot_native_config.radio_config.radio_mode = RADIO_MODE_NATIVE; - ot_native_config.host_config.host_connection_mode = HOST_CONNECTION_MODE_NONE; - ot_native_config.port_config.storage_partition_name = "nvs"; - ot_native_config.port_config.netif_queue_size = 10; - ot_native_config.port_config.task_queue_size = 10; //sTxString = ""; } @@ -256,79 +230,19 @@ OpenThreadCLI::~OpenThreadCLI() { } OpenThreadCLI::operator bool() const { - return otStarted; + return otCLIStarted; } -static void ot_task_worker(void *aContext) { - esp_vfs_eventfd_config_t eventfd_config = { - .max_fds = 3, - }; - bool err = false; - if (ESP_OK != esp_event_loop_create_default()) { - log_e("Failed to create OpentThread event loop"); - err = true; - } - if (!err && ESP_OK != esp_netif_init()) { - log_e("Failed to initialize OpentThread netif"); - err = true; - } - if (!err && ESP_OK != esp_vfs_eventfd_register(&eventfd_config)) { - log_e("Failed to register OpentThread eventfd"); - err = true; - } - - // Initialize the OpenThread stack - if (!err && ESP_OK != esp_openthread_init(&ot_native_config)) { - log_e("Failed to initialize OpenThread stack"); - err = true; - } - if (!err) { - // Initialize the OpenThread cli - otCliInit(esp_openthread_get_instance(), ot_cli_output_callback, NULL); - - // Initialize the esp_netif bindings - esp_netif_config_t cfg = ESP_NETIF_DEFAULT_OPENTHREAD(); - openthread_netif = esp_netif_new(&cfg); -#if CONFIG_LWIP_HOOK_IP6_INPUT_CUSTOM - // Get LwIP Netif - if (openthread_netif != NULL) { - ot_lwip_netif = (struct netif *)esp_netif_get_netif_impl(openthread_netif); - if (ot_lwip_netif == NULL) { - log_e("Failed to get OpenThread LwIP netif"); - } - } -#endif - } - if (!err && openthread_netif == NULL) { - log_e("Failed to create OpenThread esp_netif"); - err = true; - } - if (!err && ESP_OK != esp_netif_attach(openthread_netif, esp_openthread_netif_glue_init(&ot_native_config))) { - log_e("Failed to attach OpenThread esp_netif"); - err = true; - } - if (!err && ESP_OK != esp_netif_set_default_netif(openthread_netif)) { - log_e("Failed to set default OpenThread esp_netif"); - err = true; - } - if (!err) { - // only returns in case there is an OpenThread Stack failure... - esp_openthread_launch_mainloop(); - } - // Clean up - esp_openthread_netif_glue_deinit(); - esp_netif_destroy(openthread_netif); - esp_vfs_eventfd_unregister(); - vTaskDelete(NULL); -} - -void OpenThreadCLI::begin(bool OThreadAutoStart) { - if (otStarted) { +void OpenThreadCLI::begin() { + if (otCLIStarted) { log_w("OpenThread CLI already started. Please end() it before starting again."); return; } - xTaskCreate(ot_task_worker, "ot_main_loop", 10240, NULL, 20, &s_ot_task); + if (!OpenThread::otStarted) { + log_w("OpenThread not started. Please begin() it before starting CLI."); + return; + } //RX Buffer default has 1024 bytes if not preset if (rx_queue == NULL) { @@ -342,55 +256,29 @@ void OpenThreadCLI::begin(bool OThreadAutoStart) { log_e("HW CDC RX Buffer error"); } } + xTaskCreate(ot_cli_loop, "ot_cli", 4096, xTaskGetCurrentTaskHandle(), 2, &s_cli_task); + // Initialize the OpenThread cli + otCliInit(esp_openthread_get_instance(), ot_cli_output_callback, NULL); - // starts Thread with default dataset from NVS or from IDF default settings - if (OThreadAutoStart) { - otOperationalDatasetTlvs dataset; - otError error = otDatasetGetActiveTlvs(esp_openthread_get_instance(), &dataset); - // error = OT_ERROR_FAILED; // teste para forçar NULL dataset - if (error != OT_ERROR_NONE) { - log_i("Failed to get active NVS dataset from OpenThread"); - } else { - log_i("Got active NVS dataset from OpenThread"); - } - esp_err_t err = esp_openthread_auto_start((error == OT_ERROR_NONE) ? &dataset : NULL); - if (err != ESP_OK) { - log_i("Failed to AUTO start OpenThread"); - } else { - log_i("AUTO start OpenThread done"); - } - } - otStarted = true; + otCLIStarted = true; return; } void OpenThreadCLI::end() { - if (!otStarted) { + if (!otCLIStarted) { log_w("OpenThread CLI already stopped. Please begin() it before stopping again."); return; } - if (s_ot_task != NULL) { - vTaskDelete(s_ot_task); - // Clean up - esp_openthread_deinit(); - esp_openthread_netif_glue_deinit(); -#if CONFIG_LWIP_HOOK_IP6_INPUT_CUSTOM - ot_lwip_netif = NULL; -#endif - esp_netif_destroy(openthread_netif); - esp_vfs_eventfd_unregister(); - } if (s_cli_task != NULL) { vTaskDelete(s_cli_task); + s_cli_task = NULL; } - if (s_console_cli_task != NULL) { - vTaskDelete(s_console_cli_task); - } + stopConsole(); esp_event_loop_delete_default(); setRxBufferSize(0); setTxBufferSize(0); - otStarted = false; + otCLIStarted = false; } size_t OpenThreadCLI::write(uint8_t c) { diff --git a/libraries/OpenThread/src/OThreadCLI.h b/libraries/OpenThread/src/OThreadCLI.h index bc8dc5d2b19..788edc2709b 100644 --- a/libraries/OpenThread/src/OThreadCLI.h +++ b/libraries/OpenThread/src/OThreadCLI.h @@ -21,7 +21,6 @@ #include "esp_openthread.h" #include "esp_openthread_cli.h" #include "esp_openthread_lock.h" -#include "esp_openthread_netif_glue.h" #include "esp_openthread_types.h" #include "openthread/cli.h" @@ -31,13 +30,14 @@ #include "openthread/dataset_ftd.h" #include "Arduino.h" +#include "OThread.h" typedef std::function OnReceiveCb_t; class OpenThreadCLI : public Stream { private: static size_t setBuffer(QueueHandle_t &queue, size_t len); - bool otStarted = false; + static bool otCLIStarted; public: OpenThreadCLI(); @@ -53,7 +53,7 @@ class OpenThreadCLI : public Stream { void setStream(Stream &otStream); // changes the console Stream object void onReceive(OnReceiveCb_t func); // called on a complete line of output from OT CLI, as OT Response - void begin(bool OThreadAutoStart = true); + void begin(); void end(); // default size is 256 bytes diff --git a/libraries/OpenThread/src/OThreadCLI_Util.cpp b/libraries/OpenThread/src/OThreadCLI_Util.cpp index d21daa1effc..aad435107bb 100644 --- a/libraries/OpenThread/src/OThreadCLI_Util.cpp +++ b/libraries/OpenThread/src/OThreadCLI_Util.cpp @@ -19,26 +19,6 @@ #include "OThreadCLI_Util.h" #include -static const char *otRoleString[] = { - "Disabled", ///< The Thread stack is disabled. - "Detached", ///< Not currently participating in a Thread network/partition. - "Child", ///< The Thread Child role. - "Router", ///< The Thread Router role. - "Leader", ///< The Thread Leader role. -}; - -ot_device_role_t otGetDeviceRole() { - if (!OThreadCLI) { - return OT_ROLE_DISABLED; - } - otInstance *instance = esp_openthread_get_instance(); - return (ot_device_role_t)otThreadGetDeviceRole(instance); -} - -const char *otGetStringDeviceRole() { - return otRoleString[otGetDeviceRole()]; -} - bool otGetRespCmd(const char *cmd, char *resp, uint32_t respTimeout) { if (!OThreadCLI) { return false; @@ -174,7 +154,7 @@ bool otPrintRespCLI(const char *cmd, Stream &output, uint32_t respTimeout) { return true; } -void otPrintNetworkInformation(Stream &output) { +void otCLIPrintNetworkInformation(Stream &output) { if (!OThreadCLI) { return; } diff --git a/libraries/OpenThread/src/OThreadCLI_Util.h b/libraries/OpenThread/src/OThreadCLI_Util.h index 1ab2e061dfc..116b886e095 100644 --- a/libraries/OpenThread/src/OThreadCLI_Util.h +++ b/libraries/OpenThread/src/OThreadCLI_Util.h @@ -18,25 +18,14 @@ #if SOC_IEEE802154_SUPPORTED #if CONFIG_OPENTHREAD_ENABLED -typedef enum { - OT_ROLE_DISABLED = 0, ///< The Thread stack is disabled. - OT_ROLE_DETACHED = 1, ///< Not currently participating in a Thread network/partition. - OT_ROLE_CHILD = 2, ///< The Thread Child role. - OT_ROLE_ROUTER = 3, ///< The Thread Router role. - OT_ROLE_LEADER = 4, ///< The Thread Leader role. -} ot_device_role_t; - typedef struct { int errorCode; String errorMessage; } ot_cmd_return_t; -ot_device_role_t otGetDeviceRole(); -const char *otGetStringDeviceRole(); bool otGetRespCmd(const char *cmd, char *resp = NULL, uint32_t respTimeout = 5000); bool otExecCommand(const char *cmd, const char *arg, ot_cmd_return_t *returnCode = NULL); bool otPrintRespCLI(const char *cmd, Stream &output, uint32_t respTimeout); -void otPrintNetworkInformation(Stream &output); #endif /* CONFIG_OPENTHREAD_ENABLED */ #endif /* SOC_IEEE802154_SUPPORTED */ From 7462b09bb4818fd77a40bcaace5663e255bae8f8 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jan=20Proch=C3=A1zka?= <90197375+P-R-O-C-H-Y@users.noreply.github.com> Date: Mon, 16 Jun 2025 13:44:11 +0200 Subject: [PATCH 051/173] feat(LEDC): Add Gamma Fade support and enhance auto channel/timer selection for multi-group (#11464) * feat(ledc): Enhance LEDC auto channel/timer selection for multi-group support * feat(ledc): Add Gamma Fade support * fix(example): Update comments * ci(pre-commit): Apply automatic fixes --------- Co-authored-by: pre-commit-ci-lite[bot] <117423508+pre-commit-ci-lite[bot]@users.noreply.github.com> --- cores/esp32/esp32-hal-ledc.c | 264 ++++++++++++++++-- cores/esp32/esp32-hal-ledc.h | 79 ++++++ .../AnalogOut/LEDCGammaFade/LEDCGammaFade.ino | 111 ++++++++ .../examples/AnalogOut/LEDCGammaFade/ci.json | 5 + 4 files changed, 428 insertions(+), 31 deletions(-) create mode 100644 libraries/ESP32/examples/AnalogOut/LEDCGammaFade/LEDCGammaFade.ino create mode 100644 libraries/ESP32/examples/AnalogOut/LEDCGammaFade/ci.json diff --git a/cores/esp32/esp32-hal-ledc.c b/cores/esp32/esp32-hal-ledc.c index 764c2803b4b..18b722f80a2 100644 --- a/cores/esp32/esp32-hal-ledc.c +++ b/cores/esp32/esp32-hal-ledc.c @@ -22,6 +22,9 @@ #include "soc/gpio_sig_map.h" #include "esp_rom_gpio.h" #include "hal/ledc_ll.h" +#if SOC_LEDC_GAMMA_CURVE_FADE_SUPPORTED +#include +#endif #ifdef SOC_LEDC_SUPPORT_HS_MODE #define LEDC_CHANNELS (SOC_LEDC_CHANNEL_NUM << 1) @@ -56,7 +59,7 @@ static bool find_matching_timer(uint8_t speed_mode, uint32_t freq, uint8_t resol peripheral_bus_type_t type = perimanGetPinBusType(i); if (type == ESP32_BUS_TYPE_LEDC) { ledc_channel_handle_t *bus = (ledc_channel_handle_t *)perimanGetPinBus(i, ESP32_BUS_TYPE_LEDC); - if (bus != NULL && (bus->channel / 8) == speed_mode && bus->freq_hz == freq && bus->channel_resolution == resolution) { + if (bus != NULL && (bus->channel / SOC_LEDC_CHANNEL_NUM) == speed_mode && bus->freq_hz == freq && bus->channel_resolution == resolution) { log_d("Found matching timer %u for freq=%u, resolution=%u", bus->timer_num, freq, resolution); *timer_num = bus->timer_num; return true; @@ -78,7 +81,7 @@ static bool find_free_timer(uint8_t speed_mode, uint8_t *timer_num) { peripheral_bus_type_t type = perimanGetPinBusType(i); if (type == ESP32_BUS_TYPE_LEDC) { ledc_channel_handle_t *bus = (ledc_channel_handle_t *)perimanGetPinBus(i, ESP32_BUS_TYPE_LEDC); - if (bus != NULL && (bus->channel / 8) == speed_mode) { + if (bus != NULL && (bus->channel / SOC_LEDC_CHANNEL_NUM) == speed_mode) { log_d("Timer %u is in use by channel %u", bus->timer_num, bus->channel); used_timers |= (1 << bus->timer_num); } @@ -110,7 +113,7 @@ static void remove_channel_from_timer(uint8_t speed_mode, uint8_t timer_num, uin peripheral_bus_type_t type = perimanGetPinBusType(i); if (type == ESP32_BUS_TYPE_LEDC) { ledc_channel_handle_t *bus = (ledc_channel_handle_t *)perimanGetPinBus(i, ESP32_BUS_TYPE_LEDC); - if (bus != NULL && (bus->channel / 8) == speed_mode && bus->timer_num == timer_num && bus->channel != channel) { + if (bus != NULL && (bus->channel / SOC_LEDC_CHANNEL_NUM) == speed_mode && bus->timer_num == timer_num && bus->channel != channel) { log_d("Timer %u is still in use by channel %u", timer_num, bus->channel); timer_in_use = true; break; @@ -168,8 +171,8 @@ static bool ledcDetachBus(void *bus) { } pinMatrixOutDetach(handle->pin, false, false); if (!channel_found) { - uint8_t group = (handle->channel / 8); - remove_channel_from_timer(group, handle->timer_num, handle->channel % 8); + uint8_t group = (handle->channel / SOC_LEDC_CHANNEL_NUM); + remove_channel_from_timer(group, handle->timer_num, handle->channel % SOC_LEDC_CHANNEL_NUM); ledc_handle.used_channels &= ~(1UL << handle->channel); } free(handle); @@ -206,13 +209,13 @@ bool ledcAttachChannel(uint8_t pin, uint32_t freq, uint8_t resolution, uint8_t c return false; } - uint8_t group = (channel / 8); + uint8_t group = (channel / SOC_LEDC_CHANNEL_NUM); uint8_t timer = 0; bool channel_used = ledc_handle.used_channels & (1UL << channel); if (channel_used) { log_i("Channel %u is already set up, given frequency and resolution will be ignored", channel); - if (ledc_set_pin(pin, group, channel % 8) != ESP_OK) { + if (ledc_set_pin(pin, group, channel % SOC_LEDC_CHANNEL_NUM) != ESP_OK) { log_e("Attaching pin to already used channel failed!"); return false; } @@ -220,7 +223,7 @@ bool ledcAttachChannel(uint8_t pin, uint32_t freq, uint8_t resolution, uint8_t c // Find a timer with matching frequency and resolution, or a free timer if (!find_matching_timer(group, freq, resolution, &timer)) { if (!find_free_timer(group, &timer)) { - log_e("No free timers available for speed mode %u", group); + log_w("No free timers available for speed mode %u", group); return false; } @@ -239,12 +242,12 @@ bool ledcAttachChannel(uint8_t pin, uint32_t freq, uint8_t resolution, uint8_t c } } - uint32_t duty = ledc_get_duty(group, (channel % 8)); + uint32_t duty = ledc_get_duty(group, (channel % SOC_LEDC_CHANNEL_NUM)); ledc_channel_config_t ledc_channel; memset((void *)&ledc_channel, 0, sizeof(ledc_channel_config_t)); ledc_channel.speed_mode = group; - ledc_channel.channel = (channel % 8); + ledc_channel.channel = (channel % SOC_LEDC_CHANNEL_NUM); ledc_channel.timer_sel = timer; ledc_channel.intr_type = LEDC_INTR_DISABLE; ledc_channel.gpio_num = pin; @@ -274,7 +277,7 @@ bool ledcAttachChannel(uint8_t pin, uint32_t freq, uint8_t resolution, uint8_t c ledc_handle.used_channels |= 1UL << channel; } - if (!perimanSetPinBus(pin, ESP32_BUS_TYPE_LEDC, (void *)handle, group, channel)) { + if (!perimanSetPinBus(pin, ESP32_BUS_TYPE_LEDC, (void *)handle, channel, timer)) { ledcDetachBus((void *)handle); return false; } @@ -291,14 +294,40 @@ bool ledcAttach(uint8_t pin, uint32_t freq, uint8_t resolution) { } uint8_t channel = __builtin_ctz(free_channel); // Convert the free_channel bit to channel number - return ledcAttachChannel(pin, freq, resolution, channel); + // Try the first available channel + if (ledcAttachChannel(pin, freq, resolution, channel)) { + return true; + } + +#ifdef SOC_LEDC_SUPPORT_HS_MODE + // If first attempt failed and HS mode is supported, try to find a free channel in group 1 + if ((channel / SOC_LEDC_CHANNEL_NUM) == 0) { // First attempt was in group 0 + log_d("LEDC: Group 0 channel %u failed, trying to find a free channel in group 1", channel); + // Find free channels specifically in group 1 + uint32_t group1_mask = ((1UL << SOC_LEDC_CHANNEL_NUM) - 1) << SOC_LEDC_CHANNEL_NUM; + int group1_free_channel = (~ledc_handle.used_channels) & group1_mask; + if (group1_free_channel != 0) { + uint8_t group1_channel = __builtin_ctz(group1_free_channel); + if (ledcAttachChannel(pin, freq, resolution, group1_channel)) { + return true; + } + } + } +#endif + + log_e( + "No free timers available for freq=%u, resolution=%u. To attach a new channel, use the same frequency and resolution as an already attached channel to " + "share its timer.", + freq, resolution + ); + return false; } bool ledcWrite(uint8_t pin, uint32_t duty) { ledc_channel_handle_t *bus = (ledc_channel_handle_t *)perimanGetPinBus(pin, ESP32_BUS_TYPE_LEDC); if (bus != NULL) { - uint8_t group = (bus->channel / 8), channel = (bus->channel % 8); + uint8_t group = (bus->channel / SOC_LEDC_CHANNEL_NUM), channel = (bus->channel % SOC_LEDC_CHANNEL_NUM); //Fixing if all bits in resolution is set = LEDC FULL ON uint32_t max_duty = (1 << bus->channel_resolution) - 1; @@ -307,8 +336,14 @@ bool ledcWrite(uint8_t pin, uint32_t duty) { duty = max_duty + 1; } - ledc_set_duty(group, channel, duty); - ledc_update_duty(group, channel); + if (ledc_set_duty(group, channel, duty) != ESP_OK) { + log_e("ledc_set_duty failed"); + return false; + } + if (ledc_update_duty(group, channel) != ESP_OK) { + log_e("ledc_update_duty failed"); + return false; + } return true; } @@ -321,7 +356,11 @@ bool ledcWriteChannel(uint8_t channel, uint32_t duty) { log_e("Channel %u is not available (maximum %u) or not used!", channel, LEDC_CHANNELS); return false; } - uint8_t group = (channel / 8), timer = ((channel / 2) % 4); + uint8_t group = (channel / SOC_LEDC_CHANNEL_NUM); + ledc_timer_t timer; + + // Get the actual timer being used by this channel + ledc_ll_get_channel_timer(LEDC_LL_GET_HW(), group, (channel % SOC_LEDC_CHANNEL_NUM), &timer); //Fixing if all bits in resolution is set = LEDC FULL ON uint32_t resolution = 0; @@ -333,8 +372,14 @@ bool ledcWriteChannel(uint8_t channel, uint32_t duty) { duty = max_duty + 1; } - ledc_set_duty(group, channel, duty); - ledc_update_duty(group, channel); + if (ledc_set_duty(group, channel, duty) != ESP_OK) { + log_e("ledc_set_duty failed"); + return false; + } + if (ledc_update_duty(group, channel) != ESP_OK) { + log_e("ledc_update_duty failed"); + return false; + } return true; } @@ -343,7 +388,7 @@ uint32_t ledcRead(uint8_t pin) { ledc_channel_handle_t *bus = (ledc_channel_handle_t *)perimanGetPinBus(pin, ESP32_BUS_TYPE_LEDC); if (bus != NULL) { - uint8_t group = (bus->channel / 8), channel = (bus->channel % 8); + uint8_t group = (bus->channel / SOC_LEDC_CHANNEL_NUM), channel = (bus->channel % SOC_LEDC_CHANNEL_NUM); return ledc_get_duty(group, channel); } return 0; @@ -355,8 +400,8 @@ uint32_t ledcReadFreq(uint8_t pin) { if (!ledcRead(pin)) { return 0; } - uint8_t group = (bus->channel / 8), timer = ((bus->channel / 2) % 4); - return ledc_get_freq(group, timer); + uint8_t group = (bus->channel / SOC_LEDC_CHANNEL_NUM); + return ledc_get_freq(group, bus->timer_num); } return 0; } @@ -370,12 +415,12 @@ uint32_t ledcWriteTone(uint8_t pin, uint32_t freq) { return 0; } - uint8_t group = (bus->channel / 8), timer = ((bus->channel / 2) % 4); + uint8_t group = (bus->channel / SOC_LEDC_CHANNEL_NUM); ledc_timer_config_t ledc_timer; memset((void *)&ledc_timer, 0, sizeof(ledc_timer_config_t)); ledc_timer.speed_mode = group; - ledc_timer.timer_num = timer; + ledc_timer.timer_num = bus->timer_num; ledc_timer.duty_resolution = 10; ledc_timer.freq_hz = freq; ledc_timer.clk_cfg = clock_source; @@ -386,7 +431,7 @@ uint32_t ledcWriteTone(uint8_t pin, uint32_t freq) { } bus->channel_resolution = 10; - uint32_t res_freq = ledc_get_freq(group, timer); + uint32_t res_freq = ledc_get_freq(group, bus->timer_num); ledcWrite(pin, 0x1FF); return res_freq; } @@ -427,12 +472,12 @@ uint32_t ledcChangeFrequency(uint8_t pin, uint32_t freq, uint8_t resolution) { log_e("LEDC pin %u - resolution is zero or it is too big (maximum %u)", pin, LEDC_MAX_BIT_WIDTH); return 0; } - uint8_t group = (bus->channel / 8), timer = ((bus->channel / 2) % 4); + uint8_t group = (bus->channel / SOC_LEDC_CHANNEL_NUM); ledc_timer_config_t ledc_timer; memset((void *)&ledc_timer, 0, sizeof(ledc_timer_config_t)); ledc_timer.speed_mode = group; - ledc_timer.timer_num = timer; + ledc_timer.timer_num = bus->timer_num; ledc_timer.duty_resolution = resolution; ledc_timer.freq_hz = freq; ledc_timer.clk_cfg = clock_source; @@ -442,7 +487,7 @@ uint32_t ledcChangeFrequency(uint8_t pin, uint32_t freq, uint8_t resolution) { return 0; } bus->channel_resolution = resolution; - return ledc_get_freq(group, timer); + return ledc_get_freq(group, bus->timer_num); } return 0; } @@ -453,12 +498,14 @@ bool ledcOutputInvert(uint8_t pin, bool out_invert) { gpio_set_level(pin, out_invert); #ifdef CONFIG_IDF_TARGET_ESP32P4 - esp_rom_gpio_connect_out_signal(pin, LEDC_LS_SIG_OUT_PAD_OUT0_IDX + ((bus->channel) % 8), out_invert, 0); + esp_rom_gpio_connect_out_signal(pin, LEDC_LS_SIG_OUT_PAD_OUT0_IDX + ((bus->channel) % SOC_LEDC_CHANNEL_NUM), out_invert, 0); #else #ifdef SOC_LEDC_SUPPORT_HS_MODE - esp_rom_gpio_connect_out_signal(pin, ((bus->channel / 8 == 0) ? LEDC_HS_SIG_OUT0_IDX : LEDC_LS_SIG_OUT0_IDX) + ((bus->channel) % 8), out_invert, 0); + esp_rom_gpio_connect_out_signal( + pin, ((bus->channel / SOC_LEDC_CHANNEL_NUM == 0) ? LEDC_HS_SIG_OUT0_IDX : LEDC_LS_SIG_OUT0_IDX) + ((bus->channel) % SOC_LEDC_CHANNEL_NUM), out_invert, 0 + ); #else - esp_rom_gpio_connect_out_signal(pin, LEDC_LS_SIG_OUT0_IDX + ((bus->channel) % 8), out_invert, 0); + esp_rom_gpio_connect_out_signal(pin, LEDC_LS_SIG_OUT0_IDX + ((bus->channel) % SOC_LEDC_CHANNEL_NUM), out_invert, 0); #endif #endif // ifdef CONFIG_IDF_TARGET_ESP32P4 return true; @@ -505,7 +552,7 @@ static bool ledcFadeConfig(uint8_t pin, uint32_t start_duty, uint32_t target_dut } #endif #endif - uint8_t group = (bus->channel / 8), channel = (bus->channel % 8); + uint8_t group = (bus->channel / SOC_LEDC_CHANNEL_NUM), channel = (bus->channel % SOC_LEDC_CHANNEL_NUM); // Initialize fade service. if (!fade_initialized) { @@ -562,6 +609,161 @@ bool ledcFadeWithInterruptArg(uint8_t pin, uint32_t start_duty, uint32_t target_ return ledcFadeConfig(pin, start_duty, target_duty, max_fade_time_ms, userFunc, arg); } +#ifdef SOC_LEDC_GAMMA_CURVE_FADE_SUPPORTED +// Default gamma factor for gamma correction (common value for LEDs) +static float ledcGammaFactor = 2.8; +// Gamma correction LUT support +static const float *ledcGammaLUT = NULL; +static uint16_t ledcGammaLUTSize = 0; +// Global variable to store current resolution for gamma callback +static uint8_t ledcGammaResolution = 13; + +bool ledcSetGammaTable(const float *gamma_table, uint16_t size) { + if (gamma_table == NULL || size == 0) { + log_e("Invalid gamma table or size"); + return false; + } + ledcGammaLUT = gamma_table; + ledcGammaLUTSize = size; + log_i("Custom gamma LUT set with %u entries", size); + return true; +} + +void ledcClearGammaTable(void) { + ledcGammaLUT = NULL; + ledcGammaLUTSize = 0; + log_i("Gamma LUT cleared, using mathematical calculation"); +} + +void ledcSetGammaFactor(float factor) { + ledcGammaFactor = factor; +} + +// Gamma correction calculator function +static uint32_t ledcGammaCorrection(uint32_t duty) { + if (duty == 0) { + return 0; + } + + uint32_t max_duty = (1U << ledcGammaResolution) - 1; + if (duty >= (1U << ledcGammaResolution)) { + return max_duty; + } + + // Use LUT if provided, otherwise use mathematical calculation + if (ledcGammaLUT != NULL && ledcGammaLUTSize > 0) { + // LUT-based gamma correction + uint32_t lut_index = (duty * (ledcGammaLUTSize - 1)) / max_duty; + if (lut_index >= ledcGammaLUTSize) { + lut_index = ledcGammaLUTSize - 1; + } + + float corrected_normalized = ledcGammaLUT[lut_index]; + return (uint32_t)(corrected_normalized * max_duty); + } else { + // Mathematical gamma correction + double normalized = (double)duty / (1U << ledcGammaResolution); + double corrected = pow(normalized, ledcGammaFactor); + return (uint32_t)(corrected * (1U << ledcGammaResolution)); + } +} + +static bool ledcFadeGammaConfig(uint8_t pin, uint32_t start_duty, uint32_t target_duty, int max_fade_time_ms, void (*userFunc)(void *), void *arg) { + ledc_channel_handle_t *bus = (ledc_channel_handle_t *)perimanGetPinBus(pin, ESP32_BUS_TYPE_LEDC); + if (bus != NULL) { + +#ifndef SOC_LEDC_SUPPORT_FADE_STOP +#if !CONFIG_DISABLE_HAL_LOCKS + if (bus->lock == NULL) { + bus->lock = xSemaphoreCreateBinary(); + if (bus->lock == NULL) { + log_e("xSemaphoreCreateBinary failed"); + return false; + } + xSemaphoreGive(bus->lock); + } + //acquire lock + if (xSemaphoreTake(bus->lock, 0) != pdTRUE) { + log_e("LEDC Fade is still running on pin %u! SoC does not support stopping fade.", pin); + return false; + } +#endif +#endif + uint8_t group = (bus->channel / SOC_LEDC_CHANNEL_NUM), channel = (bus->channel % SOC_LEDC_CHANNEL_NUM); + + // Initialize fade service. + if (!fade_initialized) { + ledc_fade_func_install(0); + fade_initialized = true; + } + + bus->fn = (voidFuncPtr)userFunc; + bus->arg = arg; + + ledc_cbs_t callbacks = {.fade_cb = ledcFnWrapper}; + ledc_cb_register(group, channel, &callbacks, (void *)bus); + + // Prepare gamma curve fade parameters + ledc_fade_param_config_t fade_params[SOC_LEDC_GAMMA_CURVE_FADE_RANGE_MAX]; + uint32_t actual_fade_ranges = 0; + + // Use a moderate number of linear segments for smooth gamma curve + const uint32_t linear_fade_segments = 12; + + // Set the global resolution for gamma correction + ledcGammaResolution = bus->channel_resolution; + + // Fill multi-fade parameter list using ESP-IDF API + esp_err_t err = ledc_fill_multi_fade_param_list( + group, channel, start_duty, target_duty, linear_fade_segments, max_fade_time_ms, ledcGammaCorrection, SOC_LEDC_GAMMA_CURVE_FADE_RANGE_MAX, fade_params, + &actual_fade_ranges + ); + + if (err != ESP_OK) { + log_e("ledc_fill_multi_fade_param_list failed: %s", esp_err_to_name(err)); + return false; + } + + // Apply the gamma-corrected start duty + uint32_t gamma_start_duty = ledcGammaCorrection(start_duty); + + // Set multi-fade parameters + err = ledc_set_multi_fade(group, channel, gamma_start_duty, fade_params, actual_fade_ranges); + if (err != ESP_OK) { + log_e("ledc_set_multi_fade failed: %s", esp_err_to_name(err)); + return false; + } + + // Start the gamma curve fade + err = ledc_fade_start(group, channel, LEDC_FADE_NO_WAIT); + if (err != ESP_OK) { + log_e("ledc_fade_start failed: %s", esp_err_to_name(err)); + return false; + } + + log_d("Gamma curve fade started on pin %u: %u -> %u over %dms", pin, start_duty, target_duty, max_fade_time_ms); + + } else { + log_e("Pin %u is not attached to LEDC. Call ledcAttach first!", pin); + return false; + } + return true; +} + +bool ledcFadeGamma(uint8_t pin, uint32_t start_duty, uint32_t target_duty, int max_fade_time_ms) { + return ledcFadeGammaConfig(pin, start_duty, target_duty, max_fade_time_ms, NULL, NULL); +} + +bool ledcFadeGammaWithInterrupt(uint8_t pin, uint32_t start_duty, uint32_t target_duty, int max_fade_time_ms, voidFuncPtr userFunc) { + return ledcFadeGammaConfig(pin, start_duty, target_duty, max_fade_time_ms, (voidFuncPtrArg)userFunc, NULL); +} + +bool ledcFadeGammaWithInterruptArg(uint8_t pin, uint32_t start_duty, uint32_t target_duty, int max_fade_time_ms, void (*userFunc)(void *), void *arg) { + return ledcFadeGammaConfig(pin, start_duty, target_duty, max_fade_time_ms, userFunc, arg); +} + +#endif /* SOC_LEDC_GAMMA_CURVE_FADE_SUPPORTED */ + static uint8_t analog_resolution = 8; static int analog_frequency = 1000; void analogWrite(uint8_t pin, int value) { diff --git a/cores/esp32/esp32-hal-ledc.h b/cores/esp32/esp32-hal-ledc.h index f1a27dd4f7a..1663b884a3f 100644 --- a/cores/esp32/esp32-hal-ledc.h +++ b/cores/esp32/esp32-hal-ledc.h @@ -232,6 +232,85 @@ bool ledcFadeWithInterrupt(uint8_t pin, uint32_t start_duty, uint32_t target_dut */ bool ledcFadeWithInterruptArg(uint8_t pin, uint32_t start_duty, uint32_t target_duty, int max_fade_time_ms, void (*userFunc)(void *), void *arg); +//Gamma Curve Fade functions - only available on supported chips +#ifdef SOC_LEDC_GAMMA_CURVE_FADE_SUPPORTED + +/** + * @brief Set a custom gamma correction lookup table for gamma curve fading. + * The LUT should contain normalized values (0.0 to 1.0) representing + * the gamma-corrected brightness curve. + * + * @param gamma_table Pointer to array of float values (0.0 to 1.0) + * @param size Number of entries in the lookup table + * + * @return true if gamma table was successfully set, false otherwise. + * + * @note The LUT array must remain valid for as long as gamma fading is used. + * Larger tables provide smoother transitions but use more memory. + */ +bool ledcSetGammaTable(const float *gamma_table, uint16_t size); + +/** + * @brief Clear the current gamma correction lookup table. + * After calling this, gamma correction will use mathematical + * calculation with the default gamma factor (2.8). + */ +void ledcClearGammaTable(void); + +/** + * @brief Set the gamma factor for gamma correction. + * + * @param factor Gamma factor to use for gamma correction. + */ +void ledcSetGammaFactor(float factor); + +/** + * @brief Setup and start a gamma curve fade on a given LEDC pin. + * Gamma correction makes LED brightness changes appear more gradual to human eyes. + * + * @param pin GPIO pin + * @param start_duty initial duty cycle of the fade + * @param target_duty target duty cycle of the fade + * @param max_fade_time_ms maximum fade time in milliseconds + * + * @return true if gamma fade was successfully set and started, false otherwise. + * + * @note This function is only available on ESP32 variants that support gamma curve fading. + */ +bool ledcFadeGamma(uint8_t pin, uint32_t start_duty, uint32_t target_duty, int max_fade_time_ms); + +/** + * @brief Setup and start a gamma curve fade on a given LEDC pin with a callback function. + * + * @param pin GPIO pin + * @param start_duty initial duty cycle of the fade + * @param target_duty target duty cycle of the fade + * @param max_fade_time_ms maximum fade time in milliseconds + * @param userFunc callback function to be called after fade is finished + * + * @return true if gamma fade was successfully set and started, false otherwise. + * + * @note This function is only available on ESP32 variants that support gamma curve fading. + */ +bool ledcFadeGammaWithInterrupt(uint8_t pin, uint32_t start_duty, uint32_t target_duty, int max_fade_time_ms, void (*userFunc)(void)); + +/** + * @brief Setup and start a gamma curve fade on a given LEDC pin with a callback function and argument. + * + * @param pin GPIO pin + * @param start_duty initial duty cycle of the fade + * @param target_duty target duty cycle of the fade + * @param max_fade_time_ms maximum fade time in milliseconds + * @param userFunc callback function to be called after fade is finished + * @param arg argument to be passed to the callback function + * + * @return true if gamma fade was successfully set and started, false otherwise. + * + * @note This function is only available on ESP32 variants that support gamma curve fading. + */ +bool ledcFadeGammaWithInterruptArg(uint8_t pin, uint32_t start_duty, uint32_t target_duty, int max_fade_time_ms, void (*userFunc)(void *), void *arg); +#endif // SOC_LEDC_GAMMA_CURVE_FADE_SUPPORTED + #ifdef __cplusplus } #endif diff --git a/libraries/ESP32/examples/AnalogOut/LEDCGammaFade/LEDCGammaFade.ino b/libraries/ESP32/examples/AnalogOut/LEDCGammaFade/LEDCGammaFade.ino new file mode 100644 index 00000000000..4ca6c136ddf --- /dev/null +++ b/libraries/ESP32/examples/AnalogOut/LEDCGammaFade/LEDCGammaFade.ino @@ -0,0 +1,111 @@ +/* LEDC Gamma Curve Fade Arduino Example + + This example demonstrates gamma curve fading on ESP32 variants that support it. + Gamma correction makes LED brightness changes appear more gradual and natural + to human eyes compared to linear fading. + + Two methods are supported: + 1. Using a pre-computed Gamma Look-Up Table (LUT) for better performance + 2. Using mathematical gamma correction with a gamma factor + + Supported chips: ESP32-C6, ESP32-C5, ESP32-H2, ESP32-P4 and future chips with Gamma Fade support + + Created by Jan Procházka (https://github.com/P-R-O-C-H-Y/) +*/ + +// use 12 bit precision for LEDC timer +#define LEDC_TIMER_12_BIT 12 + +// use 5000 Hz as a LEDC base frequency +#define LEDC_BASE_FREQ 5000 + +// define starting duty, target duty and maximum fade time +#define LEDC_START_DUTY (0) +#define LEDC_TARGET_DUTY (4095) +#define LEDC_FADE_TIME (2000) + +// gamma factor for mathematical calculation +#define LEDC_GAMMA_FACTOR (2.6) + +// use gamma LUT for better performance instead of mathematical calculation (gamma factor) +#define USE_GAMMA_LUT 1 + +// fade LED pins +const uint8_t ledPinR = 4; +const uint8_t ledPinG = 5; +const uint8_t ledPinB = 6; + +uint8_t fade_ended = 0; // status of LED gamma fade +bool fade_in = true; + +#ifdef USE_GAMMA_LUT +// Custom Gamma LUT demonstration with 101 steps (Brightness 0 - 100% gamma correction look up table (gamma = 2.6)) +// Y = B ^ 2.6 - Pre-computed LUT to save runtime computation +static const float ledcGammaLUT[101] = { + 0.000000, 0.000006, 0.000038, 0.000110, 0.000232, 0.000414, 0.000666, 0.000994, 0.001406, 0.001910, 0.002512, 0.003218, 0.004035, 0.004969, 0.006025, + 0.007208, 0.008525, 0.009981, 0.011580, 0.013328, 0.015229, 0.017289, 0.019512, 0.021902, 0.024465, 0.027205, 0.030125, 0.033231, 0.036527, 0.040016, + 0.043703, 0.047593, 0.051688, 0.055993, 0.060513, 0.065249, 0.070208, 0.075392, 0.080805, 0.086451, 0.092333, 0.098455, 0.104821, 0.111434, 0.118298, + 0.125416, 0.132792, 0.140428, 0.148329, 0.156498, 0.164938, 0.173653, 0.182645, 0.191919, 0.201476, 0.211321, 0.221457, 0.231886, 0.242612, 0.253639, + 0.264968, 0.276603, 0.288548, 0.300805, 0.313378, 0.326268, 0.339480, 0.353016, 0.366879, 0.381073, 0.395599, 0.410461, 0.425662, 0.441204, 0.457091, + 0.473325, 0.489909, 0.506846, 0.524138, 0.541789, 0.559801, 0.578177, 0.596920, 0.616032, 0.635515, 0.655374, 0.675610, 0.696226, 0.717224, 0.738608, + 0.760380, 0.782542, 0.805097, 0.828048, 0.851398, 0.875148, 0.899301, 0.923861, 0.948829, 0.974208, 1.000000, +}; +#endif + +void ARDUINO_ISR_ATTR LED_FADE_ISR() { + fade_ended += 1; +} + +void setup() { + // Initialize serial communication at 115200 bits per second: + Serial.begin(115200); + + // Setup timer with given frequency, resolution and attach it to a led pin with auto-selected channel + ledcAttach(ledPinR, LEDC_BASE_FREQ, LEDC_TIMER_12_BIT); + ledcAttach(ledPinG, LEDC_BASE_FREQ, LEDC_TIMER_12_BIT); + ledcAttach(ledPinB, LEDC_BASE_FREQ, LEDC_TIMER_12_BIT); + +#if USE_GAMMA_LUT // Use default gamma LUT for better performance + ledcSetGammaTable(ledcGammaLUT, 101); +#else // Use mathematical gamma correction (default, more flexible) + ledcSetGammaFactor(LEDC_GAMMA_FACTOR); // This is optional to set custom gamma factor (default is 2.8) +#endif + + // Setup and start gamma curve fade on led (duty from 0 to 4095) + ledcFadeGamma(ledPinR, LEDC_START_DUTY, LEDC_TARGET_DUTY, LEDC_FADE_TIME); + ledcFadeGamma(ledPinG, LEDC_START_DUTY, LEDC_TARGET_DUTY, LEDC_FADE_TIME); + ledcFadeGamma(ledPinB, LEDC_START_DUTY, LEDC_TARGET_DUTY, LEDC_FADE_TIME); + Serial.println("LED Gamma Fade on started."); + + // Wait for fade to end + delay(LEDC_FADE_TIME); + + // Setup and start gamma curve fade off led and use ISR (duty from 4095 to 0) + ledcFadeGammaWithInterrupt(ledPinR, LEDC_TARGET_DUTY, LEDC_START_DUTY, LEDC_FADE_TIME, LED_FADE_ISR); + ledcFadeGammaWithInterrupt(ledPinG, LEDC_TARGET_DUTY, LEDC_START_DUTY, LEDC_FADE_TIME, LED_FADE_ISR); + ledcFadeGammaWithInterrupt(ledPinB, LEDC_TARGET_DUTY, LEDC_START_DUTY, LEDC_FADE_TIME, LED_FADE_ISR); + Serial.println("LED Gamma Fade off started."); +} + +void loop() { + // Check if fade_ended flag was set to true in ISR + if (fade_ended == 3) { + Serial.println("LED gamma fade ended"); + fade_ended = 0; + + // Check what gamma fade should be started next + if (fade_in) { + ledcFadeGammaWithInterrupt(ledPinR, LEDC_START_DUTY, LEDC_TARGET_DUTY, LEDC_FADE_TIME, LED_FADE_ISR); + ledcFadeGammaWithInterrupt(ledPinG, LEDC_START_DUTY, LEDC_TARGET_DUTY, LEDC_FADE_TIME, LED_FADE_ISR); + ledcFadeGammaWithInterrupt(ledPinB, LEDC_START_DUTY, LEDC_TARGET_DUTY, LEDC_FADE_TIME, LED_FADE_ISR); + Serial.println("LED Gamma Fade in started."); + fade_in = false; + } else { + ledcFadeGammaWithInterrupt(ledPinR, LEDC_TARGET_DUTY, LEDC_START_DUTY, LEDC_FADE_TIME, LED_FADE_ISR); + ledcFadeGammaWithInterrupt(ledPinG, LEDC_TARGET_DUTY, LEDC_START_DUTY, LEDC_FADE_TIME, LED_FADE_ISR); + ledcFadeGammaWithInterrupt(ledPinB, LEDC_TARGET_DUTY, LEDC_START_DUTY, LEDC_FADE_TIME, LED_FADE_ISR); + Serial.println("LED Gamma Fade out started."); + fade_in = true; + } + } +} diff --git a/libraries/ESP32/examples/AnalogOut/LEDCGammaFade/ci.json b/libraries/ESP32/examples/AnalogOut/LEDCGammaFade/ci.json new file mode 100644 index 00000000000..a9d8603b7bf --- /dev/null +++ b/libraries/ESP32/examples/AnalogOut/LEDCGammaFade/ci.json @@ -0,0 +1,5 @@ +{ + "requires": [ + "CONFIG_SOC_LEDC_GAMMA_CURVE_FADE_SUPPORTED=y" + ] +} From 4bc5ffc88d9246946c832d08408130d888e45b41 Mon Sep 17 00:00:00 2001 From: is-qian <2716275834@qq.com> Date: Tue, 17 Jun 2025 18:39:33 +0800 Subject: [PATCH 052/173] fix: Delete 8M flash option for xiao_esp32_s3_plus. (#11476) --- boards.txt | 6 ++---- variants/XIAO_ESP32S3_Plus/pins_arduino.h | 2 +- 2 files changed, 3 insertions(+), 5 deletions(-) diff --git a/boards.txt b/boards.txt index 15a75eaad8b..b51a8840757 100644 --- a/boards.txt +++ b/boards.txt @@ -36051,12 +36051,12 @@ XIAO_ESP32S3_Plus.build.cdc_on_boot=1 XIAO_ESP32S3_Plus.build.msc_on_boot=0 XIAO_ESP32S3_Plus.build.dfu_on_boot=0 XIAO_ESP32S3_Plus.build.f_cpu=240000000L -XIAO_ESP32S3_Plus.build.flash_size=8MB +XIAO_ESP32S3_Plus.build.flash_size=16MB XIAO_ESP32S3_Plus.build.flash_freq=80m XIAO_ESP32S3_Plus.build.flash_mode=dio XIAO_ESP32S3_Plus.build.boot=qio XIAO_ESP32S3_Plus.build.boot_freq=80m -XIAO_ESP32S3_Plus.build.partitions=default_8MB +XIAO_ESP32S3_Plus.build.partitions=ffat XIAO_ESP32S3_Plus.build.defines= XIAO_ESP32S3_Plus.build.loop_core= XIAO_ESP32S3_Plus.build.event_core= @@ -36093,8 +36093,6 @@ XIAO_ESP32S3_Plus.menu.FlashMode.dio.build.boot=dio XIAO_ESP32S3_Plus.menu.FlashMode.dio.build.boot_freq=80m XIAO_ESP32S3_Plus.menu.FlashMode.dio.build.flash_freq=80m -XIAO_ESP32S3_Plus.menu.FlashSize.8M=8MB (64Mb) -XIAO_ESP32S3_Plus.menu.FlashSize.8M.build.flash_size=8MB XIAO_ESP32S3_Plus.menu.FlashSize.16M=16MB (128Mb) XIAO_ESP32S3_Plus.menu.FlashSize.16M.build.flash_size=16MB diff --git a/variants/XIAO_ESP32S3_Plus/pins_arduino.h b/variants/XIAO_ESP32S3_Plus/pins_arduino.h index fb887287e35..de1c6093d44 100644 --- a/variants/XIAO_ESP32S3_Plus/pins_arduino.h +++ b/variants/XIAO_ESP32S3_Plus/pins_arduino.h @@ -4,7 +4,7 @@ #include #define USB_VID 0x2886 -#define USB_PID 0x0056 +#define USB_PID 0x0063 static const uint8_t LED_BUILTIN = 21; #define BUILTIN_LED LED_BUILTIN // backward compatibility From 6d4886cd1f821dbbd3827ce012dcef4e6bc14d23 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jan=20Proch=C3=A1zka?= <90197375+P-R-O-C-H-Y@users.noreply.github.com> Date: Tue, 17 Jun 2025 15:23:35 +0200 Subject: [PATCH 053/173] feat(spi): Add return values to SPI begin (#11477) --- cores/esp32/esp32-hal-spi.h | 4 ++-- libraries/SPI/src/SPI.cpp | 9 +++++---- libraries/SPI/src/SPI.h | 2 +- 3 files changed, 8 insertions(+), 7 deletions(-) diff --git a/cores/esp32/esp32-hal-spi.h b/cores/esp32/esp32-hal-spi.h index 7d56f0820d3..565ca43f60d 100644 --- a/cores/esp32/esp32-hal-spi.h +++ b/cores/esp32/esp32-hal-spi.h @@ -38,8 +38,8 @@ extern "C" { #define HSPI 2 //SPI 2 bus normally mapped to pins 12 - 15, but can be matrixed to any pins #define VSPI 3 //SPI 3 bus normally attached to pins 5, 18, 19 and 23, but can be matrixed to any pins #else -#define FSPI 0 -#define HSPI 1 +#define FSPI 0 // ESP32C2, C3, C6, H2, S3, P4 - SPI 2 bus +#define HSPI 1 // ESP32S3, P4 - SPI 3 bus #endif // This defines are not representing the real Divider of the ESP32 diff --git a/libraries/SPI/src/SPI.cpp b/libraries/SPI/src/SPI.cpp index ae207a7ff3c..c0d7665df03 100644 --- a/libraries/SPI/src/SPI.cpp +++ b/libraries/SPI/src/SPI.cpp @@ -63,9 +63,9 @@ SPIClass::~SPIClass() { #endif } -void SPIClass::begin(int8_t sck, int8_t miso, int8_t mosi, int8_t ss) { +bool SPIClass::begin(int8_t sck, int8_t miso, int8_t mosi, int8_t ss) { if (_spi) { - return; + return true; } if (!_div) { @@ -74,7 +74,7 @@ void SPIClass::begin(int8_t sck, int8_t miso, int8_t mosi, int8_t ss) { _spi = spiStartBus(_spi_num, _div, SPI_MODE0, SPI_MSBFIRST); if (!_spi) { - return; + return false; } if (sck == -1 && miso == -1 && mosi == -1 && ss == -1) { @@ -110,10 +110,11 @@ void SPIClass::begin(int8_t sck, int8_t miso, int8_t mosi, int8_t ss) { if (_mosi >= 0 && !spiAttachMOSI(_spi, _mosi)) { goto err; } - return; + return true; err: log_e("Attaching pins to SPI failed."); + return false; } void SPIClass::end() { diff --git a/libraries/SPI/src/SPI.h b/libraries/SPI/src/SPI.h index 628c2190f50..6c300e53df2 100644 --- a/libraries/SPI/src/SPI.h +++ b/libraries/SPI/src/SPI.h @@ -61,7 +61,7 @@ class SPIClass { public: SPIClass(uint8_t spi_bus = HSPI); ~SPIClass(); - void begin(int8_t sck = -1, int8_t miso = -1, int8_t mosi = -1, int8_t ss = -1); + bool begin(int8_t sck = -1, int8_t miso = -1, int8_t mosi = -1, int8_t ss = -1); void end(); void setHwCs(bool use); From f7889116b1080fbc1d802c67f795132adb69bd77 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Ludovic=20BOU=C3=89?= Date: Wed, 18 Jun 2025 19:55:11 +0200 Subject: [PATCH 054/173] feat(openthread): Add RLOC16 in otPrintNetworkInformation() (#11480) * feat(openthread): Add RLOC16 in otPrintNetworkInformation() --- libraries/OpenThread/src/OThread.cpp | 2 ++ 1 file changed, 2 insertions(+) diff --git a/libraries/OpenThread/src/OThread.cpp b/libraries/OpenThread/src/OThread.cpp index 43d8ce02918..7714d870cce 100644 --- a/libraries/OpenThread/src/OThread.cpp +++ b/libraries/OpenThread/src/OThread.cpp @@ -335,6 +335,8 @@ void OpenThread::otPrintNetworkInformation(Stream &output) { output.printf("Role: %s", otGetStringDeviceRole()); output.println(); + output.printf("RLOC16: 0x%04x", otThreadGetRloc16(mInstance)); // RLOC16 + output.println(); output.printf("Network Name: %s", otThreadGetNetworkName(mInstance)); output.println(); output.printf("Channel: %d", otLinkGetChannel(mInstance)); From 51f1367d57e07f6dd68b400771b354b501e1844f Mon Sep 17 00:00:00 2001 From: Wulu Date: Fri, 20 Jun 2025 17:22:01 +0800 Subject: [PATCH 055/173] fix(docs): correct code block indentation in core_compatibility.rst (#11471) * fix(docs): correct code block indentation in core compatibility guide * fix(docs): remove extra colon causing rendering error in core_compatibility.rst --- docs/en/guides/core_compatibility.rst | 28 +++++++++++++-------------- 1 file changed, 14 insertions(+), 14 deletions(-) diff --git a/docs/en/guides/core_compatibility.rst b/docs/en/guides/core_compatibility.rst index cb530ac5e7c..8b6a8e79d26 100644 --- a/docs/en/guides/core_compatibility.rst +++ b/docs/en/guides/core_compatibility.rst @@ -9,28 +9,28 @@ Welcome to the compatibility guide for library developers aiming to support mult Code Adaptations ---------------- -To ensure compatibility with both versions of the ESP32 Arduino core, developers should utilize conditional compilation directives in their code. Below is an example of how to conditionally include code based on the ESP32 Arduino core version:: +To ensure compatibility with both versions of the ESP32 Arduino core, developers should utilize conditional compilation directives in their code. Below is an example of how to conditionally include code based on the ESP32 Arduino core version: - .. code-block:: cpp +.. code-block:: cpp - #ifdef ESP_ARDUINO_VERSION_MAJOR - #if ESP_ARDUINO_VERSION >= ESP_ARDUINO_VERSION_VAL(3, 0, 0) - // Code for version 3.x - #else - // Code for version 2.x - #endif - #else - // Code for version 1.x - #endif + #ifdef ESP_ARDUINO_VERSION_MAJOR + #if ESP_ARDUINO_VERSION >= ESP_ARDUINO_VERSION_VAL(3, 0, 0) + // Code for version 3.x + #else + // Code for version 2.x + #endif + #else + // Code for version 1.x + #endif Version Print ------------- -To easily print the ESP32 Arduino core version at runtime, developers can use the `ESP_ARDUINO_VERSION_STR` macro. Below is an example of how to print the ESP32 Arduino core version:: +To easily print the ESP32 Arduino core version at runtime, developers can use the `ESP_ARDUINO_VERSION_STR` macro. Below is an example of how to print the ESP32 Arduino core version: - .. code-block:: cpp +.. code-block:: cpp - Serial.printf(" ESP32 Arduino core version: %s\n", ESP_ARDUINO_VERSION_STR); + Serial.printf(" ESP32 Arduino core version: %s\n", ESP_ARDUINO_VERSION_STR); API Differences --------------- From 02be6e8826fb7fad02b1d5b8492eabf541b21e03 Mon Sep 17 00:00:00 2001 From: Lucas Saavedra Vaz <32426024+lucasssvaz@users.noreply.github.com> Date: Fri, 20 Jun 2025 06:23:18 -0300 Subject: [PATCH 056/173] feat(esptool): Upgrade to esptool v5 (#11487) --- .github/scripts/package_esptool.sh | 129 ++++++++++++++++++++++ package/package_esp32_index.template.json | 70 ++++++------ platform.txt | 12 +- 3 files changed, 170 insertions(+), 41 deletions(-) create mode 100755 .github/scripts/package_esptool.sh diff --git a/.github/scripts/package_esptool.sh b/.github/scripts/package_esptool.sh new file mode 100755 index 00000000000..32b87b277e9 --- /dev/null +++ b/.github/scripts/package_esptool.sh @@ -0,0 +1,129 @@ +#!/bin/bash + +set -euo pipefail + +# Check version argument +if [[ $# -ne 3 ]]; then + echo "Usage: $0 " + echo "Example: $0 5.0.dev1 /tmp/esptool /tmp/esptool-5.0.dev1.json" + exit 1 +fi + +VERSION=$1 +BASE_FOLDER=$2 +JSON_PATH=$3 + +export COPYFILE_DISABLE=1 + +shopt -s nullglob # So for loop doesn't run if no matches + +# Function to update JSON for a given host +function update_json_for_host { + local host=$1 + local archive=$2 + + # Extract the old url from the JSON for this host, then replace only the filename + old_url=$(jq -r --arg host "$host" ' + .packages[].tools[] | select(.name == "esptool_py") | .systems[] | select(.host == $host) | .url // empty + ' "$tmp_json") + if [[ -n "$old_url" ]]; then + base_url="${old_url%/*}" + url="$base_url/$archive" + else + echo "No old url found for $host" + exit 1 + fi + + archiveFileName="$archive" + checksum="SHA-256:$(shasum -a 256 "$archive" | awk '{print $1}')" + size=$(stat -f%z "$archive") + + # Use jq to update the JSON + jq --arg host "$host" \ + --arg url "$url" \ + --arg archiveFileName "$archiveFileName" \ + --arg checksum "$checksum" \ + --arg size "$size" \ + ' + .packages[].tools[] + |= if .name == "esptool_py" then + .systems = ( + ((.systems // []) | map(select(.host != $host))) + [{ + host: $host, + url: $url, + archiveFileName: $archiveFileName, + checksum: $checksum, + size: $size + }] + ) + else + . + end + ' "$tmp_json" > "$tmp_json.new" && mv "$tmp_json.new" "$tmp_json" +} + +cd "$BASE_FOLDER" + +# Delete all archives before starting +rm -f esptool-*.tar.gz esptool-*.zip + +for dir in esptool-*; do + # Check if directory exists and is a directory + if [[ ! -d "$dir" ]]; then + continue + fi + + base="${dir#esptool-}" + + # Add 'linux-' prefix if base doesn't contain linux/macos/win64 + if [[ "$base" != *linux* && "$base" != *macos* && "$base" != *win64* ]]; then + base="linux-${base}" + fi + + if [[ "$dir" == esptool-win* ]]; then + # Windows zip archive + zipfile="esptool-v${VERSION}-${base}.zip" + echo "Creating $zipfile from $dir ..." + zip -r "$zipfile" "$dir" + else + # Non-Windows: set permissions and tar.gz archive + tarfile="esptool-v${VERSION}-${base}.tar.gz" + echo "Setting permissions and creating $tarfile from $dir ..." + chmod -R u=rwx,g=rx,o=rx "$dir" + tar -cvzf "$tarfile" "$dir" + fi +done + +# After the for loop, update the JSON for each archive +# Create a temporary JSON file to accumulate changes +tmp_json="${JSON_PATH}.tmp" +cp "$JSON_PATH" "$tmp_json" + +for archive in esptool-v"${VERSION}"-*.tar.gz esptool-v"${VERSION}"-*.zip; do + [ -f "$archive" ] || continue + + echo "Updating JSON for $archive" + + # Determine host from archive name + case "$archive" in + *linux-amd64*) host="x86_64-pc-linux-gnu" ;; + *linux-armv7*) host="arm-linux-gnueabihf" ;; + *linux-aarch64*) host="aarch64-linux-gnu" ;; + *macos-amd64*) host="x86_64-apple-darwin" ;; + *macos-arm64*) host="arm64-apple-darwin" ;; + *win64*) hosts=("x86_64-mingw32" "i686-mingw32") ;; + *) echo "Unknown host for $archive"; continue ;; + esac + + # For win64, loop over both hosts; otherwise, use a single host + if [[ "$archive" == *win64* ]]; then + for host in "${hosts[@]}"; do + update_json_for_host "$host" "$archive" + done + else + update_json_for_host "$host" "$archive" + fi +done + +# After all archives are processed, move the temporary JSON to the final file +mv "$tmp_json" "$JSON_PATH" diff --git a/package/package_esp32_index.template.json b/package/package_esp32_index.template.json index 14aa8e9b4cf..22d3cc05455 100644 --- a/package/package_esp32_index.template.json +++ b/package/package_esp32_index.template.json @@ -81,7 +81,7 @@ { "packager": "esp32", "name": "esptool_py", - "version": "4.9.dev3" + "version": "5.0.dev1" }, { "packager": "esp32", @@ -469,56 +469,56 @@ }, { "name": "esptool_py", - "version": "4.9.dev3", + "version": "5.0.dev1", "systems": [ { - "host": "x86_64-pc-linux-gnu", - "url": "https://github.com/espressif/arduino-esp32/releases/download/3.1.0-RC3/esptool-v4.9.dev3-linux-amd64.tar.gz", - "archiveFileName": "esptool-v4.9.dev3-linux-amd64.tar.gz", - "checksum": "SHA-256:4ecaf51836cbf4ea3c19840018bfef3b0b8cd8fc3c95f6e1e043ca5bbeab9bf0", - "size": "64958202" + "host": "aarch64-linux-gnu", + "url": "https://github.com/espressif/arduino-esp32/releases/download/3.2.0/esptool-v5.0.dev1-linux-aarch64.tar.gz", + "archiveFileName": "esptool-v5.0.dev1-linux-aarch64.tar.gz", + "checksum": "SHA-256:bfafa7a7723ebbabfd8b6e3ca5ae00bfead0331de923754aeddb43b2c116a078", + "size": "58241736" }, { - "host": "arm-linux-gnueabihf", - "url": "https://github.com/espressif/arduino-esp32/releases/download/3.1.0-RC3/esptool-v4.9.dev3-linux-armv7.tar.gz", - "archiveFileName": "esptool-v4.9.dev3-linux-armv7.tar.gz", - "checksum": "SHA-256:fff818573bce483ee793ac83c8211f6abf764aa3350f198228859f696a0a0b36", - "size": "31530030" + "host": "x86_64-pc-linux-gnu", + "url": "https://github.com/espressif/arduino-esp32/releases/download/3.2.0/esptool-v5.0.dev1-linux-amd64.tar.gz", + "archiveFileName": "esptool-v5.0.dev1-linux-amd64.tar.gz", + "checksum": "SHA-256:acd0486e96586b99d053a1479acbbbfcae8667227c831cdc53a171f9ccfa27ee", + "size": "100740042" }, { - "host": "aarch64-linux-gnu", - "url": "https://github.com/espressif/arduino-esp32/releases/download/3.1.0-RC3/esptool-v4.9.dev3-linux-aarch64.tar.gz", - "archiveFileName": "esptool-v4.9.dev3-linux-aarch64.tar.gz", - "checksum": "SHA-256:5b274bdff2f62e6a07c3c1dfa51b1128924621f661747eca3dbe0f77972f2f06", - "size": "33663882" + "host": "arm-linux-gnueabihf", + "url": "https://github.com/espressif/arduino-esp32/releases/download/3.2.0/esptool-v5.0.dev1-linux-armv7.tar.gz", + "archiveFileName": "esptool-v5.0.dev1-linux-armv7.tar.gz", + "checksum": "SHA-256:ea77a38681506761bbb7b0b39c130811ed565667b67ebbdb4d6dcc6cb6e07368", + "size": "53451939" }, { "host": "x86_64-apple-darwin", - "url": "https://github.com/espressif/arduino-esp32/releases/download/3.1.0-RC3/esptool-v4.9.dev3-macos-amd64.tar.gz", - "archiveFileName": "esptool-v4.9.dev3-macos-amd64.tar.gz", - "checksum": "SHA-256:c733c83b58fcf5f642fbb2fddb8ff24640c2c785126cba0821fb70c4a5ceea7a", - "size": "32767836" + "url": "https://github.com/espressif/arduino-esp32/releases/download/3.2.0/esptool-v5.0.dev1-macos-amd64.tar.gz", + "archiveFileName": "esptool-v5.0.dev1-macos-amd64.tar.gz", + "checksum": "SHA-256:900a8e90731208bee96647e0e207a43612b9452c2120c4fdc0ff4c6be226257b", + "size": "59631998" }, { "host": "arm64-apple-darwin", - "url": "https://github.com/espressif/arduino-esp32/releases/download/3.1.0-RC3/esptool-v4.9.dev3-macos-arm64.tar.gz", - "archiveFileName": "esptool-v4.9.dev3-macos-arm64.tar.gz", - "checksum": "SHA-256:83c195a15981e6a5e7a130db2ccfb21e2d8093912e5b003681f9a5abadd71af7", - "size": "30121441" + "url": "https://github.com/espressif/arduino-esp32/releases/download/3.2.0/esptool-v5.0.dev1-macos-arm64.tar.gz", + "archiveFileName": "esptool-v5.0.dev1-macos-arm64.tar.gz", + "checksum": "SHA-256:3653f4de73cb4fc6a25351eaf663708e91c65ae3265d75bd54ca4315a4350bb4", + "size": "56349992" }, { - "host": "i686-mingw32", - "url": "https://github.com/espressif/arduino-esp32/releases/download/3.1.0-RC3/esptool-v4.9.dev3-win64.zip", - "archiveFileName": "esptool-v4.9.dev3-win64.zip", - "checksum": "SHA-256:890051a4fdc684ff6f4af18d0bb27d274ca940ee0eef716a9455f8c64b25b215", - "size": "36072564" + "host": "x86_64-mingw32", + "url": "https://github.com/espressif/arduino-esp32/releases/download/3.2.0/esptool-v5.0.dev1-win64.zip", + "archiveFileName": "esptool-v5.0.dev1-win64.zip", + "checksum": "SHA-256:1e8fd89645daf94f2d4406ec73c9004e617ea921079515f9fd749205eece4d6d", + "size": "59102658" }, { - "host": "x86_64-mingw32", - "url": "https://github.com/espressif/arduino-esp32/releases/download/3.1.0-RC3/esptool-v4.9.dev3-win64.zip", - "archiveFileName": "esptool-v4.9.dev3-win64.zip", - "checksum": "SHA-256:890051a4fdc684ff6f4af18d0bb27d274ca940ee0eef716a9455f8c64b25b215", - "size": "36072564" + "host": "i686-mingw32", + "url": "https://github.com/espressif/arduino-esp32/releases/download/3.2.0/esptool-v5.0.dev1-win64.zip", + "archiveFileName": "esptool-v5.0.dev1-win64.zip", + "checksum": "SHA-256:1e8fd89645daf94f2d4406ec73c9004e617ea921079515f9fd749205eece4d6d", + "size": "59102658" } ] }, diff --git a/platform.txt b/platform.txt index 65be05b3bf4..8c7f4432377 100644 --- a/platform.txt +++ b/platform.txt @@ -119,7 +119,7 @@ recipe.hooks.prebuild.2.pattern.windows=cmd /c if not exist "{build.path}\partit recipe.hooks.prebuild.3.pattern.windows=cmd /c if not exist "{build.path}\partitions.csv" COPY "{runtime.platform.path}\tools\partitions\{build.partitions}.csv" "{build.path}\partitions.csv" # Check if custom bootloader exist: source > variant > build.boot -recipe.hooks.prebuild.4.pattern_args=--chip {build.mcu} elf2image --flash_mode {build.flash_mode} --flash_freq {build.img_freq} --flash_size {build.flash_size} -o +recipe.hooks.prebuild.4.pattern_args=--chip {build.mcu} elf2image --flash-mode {build.flash_mode} --flash-freq {build.img_freq} --flash-size {build.flash_size} -o recipe.hooks.prebuild.4.pattern=/usr/bin/env bash -c "[ -f "{build.source.path}"/bootloader.bin ] && cp -f "{build.source.path}"/bootloader.bin "{build.path}"/{build.project_name}.bootloader.bin || ( [ -f "{build.variant.path}"/{build.custom_bootloader}.bin ] && cp "{build.variant.path}"/{build.custom_bootloader}.bin "{build.path}"/{build.project_name}.bootloader.bin || "{tools.esptool_py.path}"/{tools.esptool_py.cmd} {recipe.hooks.prebuild.4.pattern_args} "{build.path}"/{build.project_name}.bootloader.bin "{compiler.sdk.path}"/bin/bootloader_{build.boot}_{build.boot_freq}.elf )" recipe.hooks.prebuild.4.pattern.windows=cmd /c IF EXIST "{build.source.path}\bootloader.bin" ( COPY /y "{build.source.path}\bootloader.bin" "{build.path}\{build.project_name}.bootloader.bin" ) ELSE ( IF EXIST "{build.variant.path}\{build.custom_bootloader}.bin" ( COPY "{build.variant.path}\{build.custom_bootloader}.bin" "{build.path}\{build.project_name}.bootloader.bin" ) ELSE ( "{tools.esptool_py.path}\{tools.esptool_py.cmd}" {recipe.hooks.prebuild.4.pattern_args} "{build.path}\{build.project_name}.bootloader.bin" "{compiler.sdk.path}\bin\bootloader_{build.boot}_{build.boot_freq}.elf" ) ) @@ -163,7 +163,7 @@ recipe.c.combine.pattern="{compiler.path}{compiler.c.elf.cmd}" {compiler.c.elf.f recipe.objcopy.partitions.bin.pattern={tools.gen_esp32part.cmd} -q "{build.path}/partitions.csv" "{build.path}/{build.project_name}.partitions.bin" ## Create bin -recipe.objcopy.bin.pattern_args=--chip {build.mcu} elf2image --flash_mode "{build.flash_mode}" --flash_freq "{build.img_freq}" --flash_size "{build.flash_size}" --elf-sha256-offset 0xb0 -o "{build.path}/{build.project_name}.bin" "{build.path}/{build.project_name}.elf" +recipe.objcopy.bin.pattern_args=--chip {build.mcu} elf2image --flash-mode "{build.flash_mode}" --flash-freq "{build.img_freq}" --flash-size "{build.flash_size}" --elf-sha256-offset 0xb0 -o "{build.path}/{build.project_name}.bin" "{build.path}/{build.project_name}.elf" recipe.objcopy.bin.pattern="{tools.esptool_py.path}/{tools.esptool_py.cmd}" {recipe.objcopy.bin.pattern_args} ## Create Insights Firmware Package @@ -176,7 +176,7 @@ recipe.hooks.objcopy.postobjcopy.2.pattern=/usr/bin/env bash -c "[ ! -d "{build. recipe.hooks.objcopy.postobjcopy.2.pattern.windows=cmd /c if exist "{build.path}\libraries\ESP_SR" if exist "{compiler.sdk.path}\esp_sr\srmodels.bin" COPY /y "{compiler.sdk.path}\esp_sr\srmodels.bin" "{build.path}\srmodels.bin" # Create merged binary -recipe.hooks.objcopy.postobjcopy.3.pattern_args=--chip {build.mcu} merge_bin -o "{build.path}/{build.project_name}.merged.bin" --fill-flash-size {build.flash_size} --flash_mode keep --flash_freq keep --flash_size keep {build.bootloader_addr} "{build.path}/{build.project_name}.bootloader.bin" 0x8000 "{build.path}/{build.project_name}.partitions.bin" 0xe000 "{runtime.platform.path}/tools/partitions/boot_app0.bin" 0x10000 "{build.path}/{build.project_name}.bin" +recipe.hooks.objcopy.postobjcopy.3.pattern_args=--chip {build.mcu} merge-bin -o "{build.path}/{build.project_name}.merged.bin" --pad-to-size {build.flash_size} --flash-mode keep --flash-freq keep --flash-size keep {build.bootloader_addr} "{build.path}/{build.project_name}.bootloader.bin" 0x8000 "{build.path}/{build.project_name}.partitions.bin" 0xe000 "{runtime.platform.path}/tools/partitions/boot_app0.bin" 0x10000 "{build.path}/{build.project_name}.bin" recipe.hooks.objcopy.postobjcopy.3.pattern="{tools.esptool_py.path}/{tools.esptool_py.cmd}" {recipe.hooks.objcopy.postobjcopy.3.pattern_args} ## Save bin @@ -285,14 +285,14 @@ debug.additional_config=debug_config.{build.mcu} tools.esptool_py.upload.protocol=serial tools.esptool_py.upload.params.verbose= tools.esptool_py.upload.params.quiet= -tools.esptool_py.upload.pattern_args=--chip {build.mcu} --port "{serial.port}" --baud {upload.speed} {upload.flags} --before default_reset --after hard_reset write_flash {upload.erase_cmd} -z --flash_mode keep --flash_freq keep --flash_size keep {build.bootloader_addr} "{build.path}/{build.project_name}.bootloader.bin" 0x8000 "{build.path}/{build.project_name}.partitions.bin" 0xe000 "{runtime.platform.path}/tools/partitions/boot_app0.bin" 0x10000 "{build.path}/{build.project_name}.bin" {upload.extra_flags} +tools.esptool_py.upload.pattern_args=--chip {build.mcu} --port "{serial.port}" --baud {upload.speed} {upload.flags} --before default-reset --after hard-reset write-flash {upload.erase_cmd} -z --flash-mode keep --flash-freq keep --flash-size keep {build.bootloader_addr} "{build.path}/{build.project_name}.bootloader.bin" 0x8000 "{build.path}/{build.project_name}.partitions.bin" 0xe000 "{runtime.platform.path}/tools/partitions/boot_app0.bin" 0x10000 "{build.path}/{build.project_name}.bin" {upload.extra_flags} tools.esptool_py.upload.pattern="{path}/{cmd}" {upload.pattern_args} ## Program Application ## ------------------- tools.esptool_py.program.params.verbose= tools.esptool_py.program.params.quiet= -tools.esptool_py.program.pattern_args=--chip {build.mcu} --port "{serial.port}" --baud {upload.speed} {upload.flags} --before default_reset --after hard_reset write_flash -z --flash_mode keep --flash_freq keep --flash_size keep 0x10000 "{build.path}/{build.project_name}.bin" +tools.esptool_py.program.pattern_args=--chip {build.mcu} --port "{serial.port}" --baud {upload.speed} {upload.flags} --before default-reset --after hard-reset write-flash -z --flash-mode keep --flash-freq keep --flash-size keep 0x10000 "{build.path}/{build.project_name}.bin" tools.esptool_py.program.pattern="{path}/{cmd}" {program.pattern_args} ## Erase Chip (before burning the bootloader) @@ -300,7 +300,7 @@ tools.esptool_py.program.pattern="{path}/{cmd}" {program.pattern_args} tools.esptool_py.erase.protocol=serial tools.esptool_py.erase.params.verbose= tools.esptool_py.erase.params.quiet= -tools.esptool_py.erase.pattern_args=--chip {build.mcu} --port "{serial.port}" --baud {upload.speed} {upload.flags} --before default_reset --after hard_reset erase_flash +tools.esptool_py.erase.pattern_args=--chip {build.mcu} --port "{serial.port}" --baud {upload.speed} {upload.flags} --before default-reset --after hard-reset erase-flash tools.esptool_py.erase.pattern="{path}/{cmd}" {erase.pattern_args} ## Burn Bootloader From 016077e2450a29bf94c681e9db23ace55a025e5f Mon Sep 17 00:00:00 2001 From: Jason2866 <24528715+Jason2866@users.noreply.github.com> Date: Fri, 20 Jun 2025 11:25:03 +0200 Subject: [PATCH 057/173] changes for updated esptool.py v5 (#11488) --- tools/pioarduino-build.py | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/tools/pioarduino-build.py b/tools/pioarduino-build.py index 3335a716888..b6a0c0435c5 100644 --- a/tools/pioarduino-build.py +++ b/tools/pioarduino-build.py @@ -99,11 +99,11 @@ def generate_bootloader_image(bootloader_elf): "--chip", build_mcu, "elf2image", - "--flash_mode", + "--flash-mode", "${__get_board_flash_mode(__env__)}", - "--flash_freq", + "--flash-freq", "${__get_board_f_image(__env__)}", - "--flash_size", + "--flash-size", board_config.get("upload.flash_size", "4MB"), "-o", "$TARGET", From 23c6abc1acd26f171827a7459572f9efc216de0e Mon Sep 17 00:00:00 2001 From: Olaf Date: Fri, 20 Jun 2025 11:52:21 +0200 Subject: [PATCH 058/173] Proper EDNS handling and cleaner NOERROR response in DNSSERVER (#11411) * Proper EDNS handling and cleaner NOERROR response * fix: library.properties reverting version number update - as it is done automatically * ci(pre-commit): Apply automatic fixes * Spelling Corrected and minor clarification in comments * Removing commented out code fragments * ci(pre-commit): Apply automatic fixes * fix(pr): Fix typo --------- Co-authored-by: pre-commit-ci-lite[bot] <117423508+pre-commit-ci-lite[bot]@users.noreply.github.com> Co-authored-by: Me No Dev --- libraries/DNSServer/src/DNSServer.cpp | 86 +++++++++++++++++++++++++-- libraries/DNSServer/src/DNSServer.h | 22 +++++++ 2 files changed, 104 insertions(+), 4 deletions(-) diff --git a/libraries/DNSServer/src/DNSServer.cpp b/libraries/DNSServer/src/DNSServer.cpp index 28cf89d6ede..1f74c96c733 100644 --- a/libraries/DNSServer/src/DNSServer.cpp +++ b/libraries/DNSServer/src/DNSServer.cpp @@ -111,16 +111,22 @@ void DNSServer::_handleUDP(AsyncUDPPacket &pkt) { // will reply with IP only to "*" or if domain matches without www. subdomain if (dnsHeader.OPCode == DNS_OPCODE_QUERY && requestIncludesOnlyOneQuestion(dnsHeader) && (_domainName.isEmpty() || getDomainNameWithoutWwwPrefix(static_cast(dnsQuestion.QName), dnsQuestion.QNameLength) == _domainName)) { - replyWithIP(pkt, dnsHeader, dnsQuestion); + + // Qtype = A (1) or ANY (255): send an A record otherwise an empty response + if (ntohs(dnsQuestion.QType) == 1 || ntohs(dnsQuestion.QType) == 255) { + replyWithIP(pkt, dnsHeader, dnsQuestion); + } else { + replyWithNoAnsw(pkt, dnsHeader, dnsQuestion); + } return; } - // otherwise reply with custom code replyWithCustomCode(pkt, dnsHeader); } bool DNSServer::requestIncludesOnlyOneQuestion(DNSHeader &dnsHeader) { - return ntohs(dnsHeader.QDCount) == 1 && dnsHeader.ANCount == 0 && dnsHeader.NSCount == 0 && dnsHeader.ARCount == 0; + dnsHeader.ARCount = 0; // We assume that if ARCount !=0 there is a EDNS OPT packet, just ignore + return ntohs(dnsHeader.QDCount) == 1 && dnsHeader.ANCount == 0 && dnsHeader.NSCount == 0; } String DNSServer::getDomainNameWithoutWwwPrefix(const unsigned char *start, size_t len) { @@ -139,7 +145,6 @@ String DNSServer::getDomainNameWithoutWwwPrefix(const unsigned char *start, size void DNSServer::replyWithIP(AsyncUDPPacket &req, DNSHeader &dnsHeader, DNSQuestion &dnsQuestion) { AsyncUDPMessage rpl; - // Change the type of message to a response and set the number of answers equal to // the number of questions in the header dnsHeader.QR = DNS_QR_RESPONSE; @@ -187,3 +192,76 @@ void DNSServer::replyWithCustomCode(AsyncUDPPacket &req, DNSHeader &dnsHeader) { rpl.write(reinterpret_cast(&dnsHeader), sizeof(DNSHeader)); _udp.sendTo(rpl, req.remoteIP(), req.remotePort()); } + +void DNSServer::replyWithNoAnsw(AsyncUDPPacket &req, DNSHeader &dnsHeader, DNSQuestion &dnsQuestion) { + + dnsHeader.QR = DNS_QR_RESPONSE; + dnsHeader.ANCount = 0; + dnsHeader.NSCount = htons(1); + + AsyncUDPMessage rpl; + rpl.write(reinterpret_cast(&dnsHeader), sizeof(DNSHeader)); + + // Write the question + rpl.write(dnsQuestion.QName, dnsQuestion.QNameLength); + rpl.write((uint8_t *)&dnsQuestion.QType, 2); + rpl.write((uint8_t *)&dnsQuestion.QClass, 2); + + // An empty answer contains an authority section with a SOA, + // We take the name of the query as the root of the zone for which the SOA is generated + // and use a value of DNS_MINIMAL_TTL seconds in order to minimize negative caching + // Write the authority section: + // The SOA RR's ownername is set equal to the query name, and we use made up names for + // the MNAME and RNAME - it doesn't really matter from a protocol perspective - as for + // a no such QTYPE answer only the timing fields are used. + // a protocol perspective - it + // Use DNS name compression : instead of repeating the name in this RNAME occurrence, + // set the two MSB of the byte corresponding normally to the length to 1. The following + // 14 bits must be used to specify the offset of the domain name in the message + // (<255 here so the first byte has the 6 LSB at 0) + rpl.write((uint8_t)0xC0); + rpl.write((uint8_t)DNS_OFFSET_DOMAIN_NAME); + + // DNS type A : host address, DNS class IN for INternet, returning an IPv4 address + uint16_t answerType = htons(DNS_TYPE_SOA), answerClass = htons(DNS_CLASS_IN); + uint32_t Serial = htonl(DNS_SOA_SERIAL); // Date type serial based on the date this piece of code was written + uint32_t Refresh = htonl(DNS_SOA_REFRESH); // These timers don't matter, we don't serve zone transfers + uint32_t Retry = htonl(DNS_SOA_RETRY); + uint32_t Expire = htonl(DNS_SOA_EXPIRE); + uint32_t MinTTL = htonl(DNS_MINIMAL_TTL); // See RFC2308 section 5 + char MLabel[] = DNS_SOA_MNAME_LABEL; + char RLabel[] = DNS_SOA_RNAME_LABEL; + char PostFixLabel[] = DNS_SOA_POSTFIX_LABEL; + + // 4 accounts for len fields and for both rname + // and lname and their postfix labels and there are 5 32 bit fields + + uint16_t RdataLength = htons((uint16_t)(strlen(MLabel) + strlen(RLabel) + 2 * strlen(PostFixLabel) + 4 + 5 * sizeof(Serial))); + + rpl.write((unsigned char *)&answerType, 2); + rpl.write((unsigned char *)&answerClass, 2); + rpl.write((unsigned char *)&MinTTL, 4); // DNS Time To Live + + rpl.write((unsigned char *)&RdataLength, 2); + + rpl.write((uint8_t)strlen(MLabel)); + rpl.write((unsigned char *)&MLabel, strlen(MLabel)); + + rpl.write((unsigned char *)&PostFixLabel, strlen(PostFixLabel)); + rpl.write((uint8_t)0); + // rpl.write((uint8_t)0xC0); + // rpl.write((uint8_t)DNS_OFFSET_DOMAIN_NAME); + + rpl.write((uint8_t)strlen(RLabel)); + rpl.write((unsigned char *)&RLabel, strlen(RLabel)); + rpl.write((unsigned char *)&PostFixLabel, strlen(PostFixLabel)); + rpl.write((uint8_t)0); + + rpl.write((unsigned char *)&Serial, 4); + rpl.write((unsigned char *)&Refresh, 4); + rpl.write((unsigned char *)&Retry, 4); + rpl.write((unsigned char *)&Expire, 4); + rpl.write((unsigned char *)&MinTTL, 4); + + _udp.sendTo(rpl, req.remoteIP(), req.remotePort()); +} diff --git a/libraries/DNSServer/src/DNSServer.h b/libraries/DNSServer/src/DNSServer.h index dfd9a45604d..f2716defc7d 100644 --- a/libraries/DNSServer/src/DNSServer.h +++ b/libraries/DNSServer/src/DNSServer.h @@ -9,6 +9,26 @@ #define DNS_OFFSET_DOMAIN_NAME DNS_HEADER_SIZE // Offset in bytes to reach the domain name labels in the DNS message #define DNS_DEFAULT_PORT 53 +#define DNS_SOA_MNAME_LABEL "ns" +#define DNS_SOA_RNAME_LABEL "esp32" +// The POSTFIX_LABEL will be concatenated to the RName and MName Label label +// do not use a multilabel name here. "local" is a good choice as it is reserved for +// local use by IANA +// The postfix label is defined as an array of characters that follows the +// definition of RFC1035 3.1 +// for instance, a postfix of example.com would be defined as: +// #define DNS_SOA_POSTFIX_LABEL {'\7', 'e', 'x', 'a', 'm', 'p', 'l', 'e', '\3', 'c', 'o', 'm', '\0'} +#define DNS_SOA_POSTFIX_LABEL \ + { '\5', 'l', 'o', 'c', 'a', 'l', '\0' } +// From the following values only the MINIMAL_TTL has relevance +// in the context of client-server protocol interactions. +// The other values are arbitrary chosen as they are only relevant for +// in a zone-transfer scenario. +#define DNS_SOA_SERIAL 2025052900 // Arbitrary serial (format: YYYYMMDDnn) +#define DNS_SOA_REFRESH 100000 // Arbitrary (seconds) +#define DNS_SOA_RETRY 10000 // Arbitrary (seconds) +#define DNS_SOA_EXPIRE 1000000 // Arbitrary (seconds) +#define DNS_MINIMAL_TTL 5 // Time to live for negative answers RFC2308 enum class DNSReplyCode : uint16_t { NoError = 0, FormError = 1, @@ -179,5 +199,7 @@ class DNSServer { inline bool requestIncludesOnlyOneQuestion(DNSHeader &dnsHeader); void replyWithIP(AsyncUDPPacket &req, DNSHeader &dnsHeader, DNSQuestion &dnsQuestion); inline void replyWithCustomCode(AsyncUDPPacket &req, DNSHeader &dnsHeader); + inline void replyWithNoAnsw(AsyncUDPPacket &req, DNSHeader &dnsHeader, DNSQuestion &dnsQuestion); + void _handleUDP(AsyncUDPPacket &pkt); }; From ccda9c5f846532972a3f497addba0623935f7cb9 Mon Sep 17 00:00:00 2001 From: Sugar Glider Date: Fri, 20 Jun 2025 15:28:07 -0300 Subject: [PATCH 059/173] feat(matter): Adds Matter Events callback plus example (#11465) * feat(matter): Adds Matter Events callback and related example --- .../examples/MatterEvents/MatterEvents.ino | 168 ++++++++++++++++++ .../Matter/examples/MatterEvents/ci.json | 7 + libraries/Matter/keywords.txt | 39 +++- libraries/Matter/src/Matter.cpp | 37 ++-- libraries/Matter/src/Matter.h | 126 +++++++++++++ libraries/Matter/src/MatterEndPoint.h | 20 +-- 6 files changed, 369 insertions(+), 28 deletions(-) create mode 100644 libraries/Matter/examples/MatterEvents/MatterEvents.ino create mode 100644 libraries/Matter/examples/MatterEvents/ci.json diff --git a/libraries/Matter/examples/MatterEvents/MatterEvents.ino b/libraries/Matter/examples/MatterEvents/MatterEvents.ino new file mode 100644 index 00000000000..dac599bf9fa --- /dev/null +++ b/libraries/Matter/examples/MatterEvents/MatterEvents.ino @@ -0,0 +1,168 @@ +// Copyright 2025 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. + +// Matter Manager +#include +#include + +// WiFi is manually set and started +const char *ssid = "your-ssid"; // Change this to your WiFi SSID +const char *password = "your-password"; // Change this to your WiFi password + +// List of Matter Endpoints for this Node +// On/Off Light Endpoint +MatterOnOffLight OnOffLight; + +// This function is called when a Matter event occurs +void onMatterEvent(matterEvent_t eventType, const chip::DeviceLayer::ChipDeviceEvent *eventInfo) { + // Print the event type to Serial + Serial.print("===> Got a Matter Event: "); + switch (eventType) { + case MATTER_WIFI_CONNECTIVITY_CHANGE: Serial.println("WiFi Connectivity Change"); break; + case MATTER_THREAD_CONNECTIVITY_CHANGE: Serial.println("Thread Connectivity Change"); break; + case MATTER_INTERNET_CONNECTIVITY_CHANGE: + { + bool newIPAddress = false; + Serial.print("Internet Connectivity Change :: "); + if (eventInfo->InternetConnectivityChange.IPv4 != chip::DeviceLayer::ConnectivityChange::kConnectivity_NoChange) { + Serial.print("IPv4 Connectivity: "); + switch (eventInfo->InternetConnectivityChange.IPv4) { + case chip::DeviceLayer::ConnectivityChange::kConnectivity_Established: + { + newIPAddress = true; + break; + } + case chip::DeviceLayer::ConnectivityChange::kConnectivity_Lost: Serial.println("Lost"); break; + default: Serial.println("Unknown"); break; + } + } + if (eventInfo->InternetConnectivityChange.IPv6 != chip::DeviceLayer::ConnectivityChange::kConnectivity_NoChange) { + Serial.print("IPv6 Connectivity: "); + switch (eventInfo->InternetConnectivityChange.IPv6) { + case chip::DeviceLayer::ConnectivityChange::kConnectivity_Established: + { + newIPAddress = true; + break; + } + case chip::DeviceLayer::ConnectivityChange::kConnectivity_Lost: Serial.println("Lost"); break; + default: Serial.println("Unknown"); break; + } + } + // Print the IP address if it was established + if (newIPAddress) { + Serial.print("Established - IP Address: "); + char ipAddressStr[chip::Transport::PeerAddress::kMaxToStringSize]; + eventInfo->InternetConnectivityChange.ipAddress.ToString(ipAddressStr); + Serial.println(ipAddressStr); + } + break; + } + case MATTER_SERVICE_CONNECTIVITY_CHANGE: Serial.println("Service Connectivity Change"); break; + case MATTER_SERVICE_PROVISIONING_CHANGE: Serial.println("Service Provisioning Change"); break; + case MATTER_TIME_SYNC_CHANGE: Serial.println("Time Sync Change"); break; + case MATTER_CHIPOBLE_CONNECTION_ESTABLISHED: Serial.println("CHIPoBLE Connection Established"); break; + case MATTER_CHIPOBLE_CONNECTION_CLOSED: Serial.println("CHIPoBLE Connection Closed"); break; + case MATTER_CLOSE_ALL_BLE_CONNECTIONS: Serial.println("Close All BLE Connections"); break; + case MATTER_WIFI_DEVICE_AVAILABLE: Serial.println("WiFi Device Available"); break; + case MATTER_OPERATIONAL_NETWORK_STARTED: Serial.println("Operational Network Started"); break; + case MATTER_THREAD_STATE_CHANGE: Serial.println("Thread State Change"); break; + case MATTER_THREAD_INTERFACE_STATE_CHANGE: Serial.println("Thread Interface State Change"); break; + case MATTER_CHIPOBLE_ADVERTISING_CHANGE: Serial.println("CHIPoBLE Advertising Change"); break; + case MATTER_INTERFACE_IP_ADDRESS_CHANGED: + switch (eventInfo->InterfaceIpAddressChanged.Type) { + case chip::DeviceLayer::InterfaceIpChangeType::kIpV4_Assigned: Serial.println("IPv4 Address Assigned"); break; + case chip::DeviceLayer::InterfaceIpChangeType::kIpV4_Lost: Serial.println("IPv4 Address Lost"); break; + case chip::DeviceLayer::InterfaceIpChangeType::kIpV6_Assigned: Serial.println("IPv6 Address Assigned"); break; + case chip::DeviceLayer::InterfaceIpChangeType::kIpV6_Lost: Serial.println("IPv6 Address Lost"); break; + } + break; + case MATTER_COMMISSIONING_COMPLETE: Serial.println("Commissioning Complete"); break; + case MATTER_FAIL_SAFE_TIMER_EXPIRED: Serial.println("Fail Safe Timer Expired"); break; + case MATTER_OPERATIONAL_NETWORK_ENABLED: Serial.println("Operational Network Enabled"); break; + case MATTER_DNSSD_INITIALIZED: Serial.println("DNS-SD Initialized"); break; + case MATTER_DNSSD_RESTART_NEEDED: Serial.println("DNS-SD Restart Needed"); break; + case MATTER_BINDINGS_CHANGED_VIA_CLUSTER: Serial.println("Bindings Changed Via Cluster"); break; + case MATTER_OTA_STATE_CHANGED: Serial.println("OTA State Changed"); break; + case MATTER_SERVER_READY: Serial.println("Server Ready"); break; + case MATTER_BLE_DEINITIALIZED: Serial.println("BLE Deinitialized"); break; + case MATTER_COMMISSIONING_SESSION_STARTED: Serial.println("Commissioning Session Started"); break; + case MATTER_COMMISSIONING_SESSION_STOPPED: Serial.println("Commissioning Session Stopped"); break; + case MATTER_COMMISSIONING_WINDOW_OPEN: Serial.println("Commissioning Window Opened"); break; + case MATTER_COMMISSIONING_WINDOW_CLOSED: Serial.println("Commissioning Window Closed"); break; + case MATTER_FABRIC_WILL_BE_REMOVED: Serial.println("Fabric Will Be Removed"); break; + case MATTER_FABRIC_REMOVED: Serial.println("Fabric Removed"); break; + case MATTER_FABRIC_COMMITTED: Serial.println("Fabric Committed"); break; + case MATTER_FABRIC_UPDATED: Serial.println("Fabric Updated"); break; + case MATTER_ESP32_SPECIFIC_EVENT: Serial.println("Sending ESP32 Platform Specific Events"); break; + case MATTER_ESP32_PUBLIC_SPECIFIC_EVENT: Serial.println("Next Event Has Populated EventInfo"); break; + default: + // If the event type is not recognized, print "Unknown" and the event ID + Serial.println("Unknown, EventID = 0x" + String(eventType, HEX)); + break; + } +} + +void setup() { + Serial.begin(115200); + while (!Serial) { + delay(10); // Wait for Serial to initialize + } + + // We start by connecting to a WiFi network + Serial.print("Connecting to "); + Serial.println(ssid); + // Manually connect to WiFi + WiFi.enableIPv6(true); // Enable IPv6 if needed + WiFi.begin(ssid, password); + // Wait for connection + while (WiFi.status() != WL_CONNECTED) { + delay(500); + Serial.print("."); + } + Serial.println("\r\nWiFi connected"); + Serial.println("IP address: "); + Serial.println(WiFi.localIP()); + delay(500); + + // Initialize at least one Matter EndPoint + OnOffLight.begin(); + + // Set the Matter Event Callback + Matter.onEvent(onMatterEvent); + // Matter beginning - Last step, after all EndPoints are initialized + Matter.begin(); + Serial.println("Starting Matter Commission Test..."); +} + +void loop() { + // Check Matter Commissioning state + if (!Matter.isDeviceCommissioned()) { + Serial.println(""); + Serial.println("Matter Node is not commissioned yet."); + Serial.println("Initiate the device discovery in your Matter environment."); + Serial.println("Commission it to your Matter hub with the manual pairing code or QR code"); + Serial.printf("Manual pairing code: %s\r\n", Matter.getManualPairingCode().c_str()); + Serial.printf("QR code URL: %s\r\n", Matter.getOnboardingQRCodeUrl().c_str()); + // waits for Matter Light Commissioning. + while (!Matter.isDeviceCommissioned()) { + delay(5000); + Serial.println("Matter Fabric not commissioned yet. Waiting for commissioning."); + } + } + Serial.println("Matter Node is commissioned and connected to Wi-Fi."); + Serial.println("====> Decommissioning in 60 seconds. <===="); + delay(60000); + Matter.decommission(); + Serial.println("Matter Node is decommissioned. Commissioning widget shall start over."); +} diff --git a/libraries/Matter/examples/MatterEvents/ci.json b/libraries/Matter/examples/MatterEvents/ci.json new file mode 100644 index 00000000000..556a8a9ee6b --- /dev/null +++ b/libraries/Matter/examples/MatterEvents/ci.json @@ -0,0 +1,7 @@ +{ + "fqbn_append": "PartitionScheme=huge_app", + "requires": [ + "CONFIG_SOC_WIFI_SUPPORTED=y", + "CONFIG_ESP_MATTER_ENABLE_DATA_MODEL=y" + ] +} diff --git a/libraries/Matter/keywords.txt b/libraries/Matter/keywords.txt index a63d9a65acb..68aaebb1d4d 100644 --- a/libraries/Matter/keywords.txt +++ b/libraries/Matter/keywords.txt @@ -36,6 +36,8 @@ EndPointSpeedCB KEYWORD1 EndPointOnOffCB KEYWORD1 EndPointBrightnessCB KEYWORD1 EndPointRGBColorCB KEYWORD1 +matterEvent_t KEYWORD1 +matterEventCB KEYWORD1 ####################################### # Methods and Functions (KEYWORD2) @@ -108,6 +110,7 @@ onChangeMode KEYWORD2 onChangeLocalTemperature KEYWORD2 onChangeCoolingSetpoint KEYWORD2 onChangeHeatingSetpoint KEYWORD2 +onEvent KEYWORD2 ####################################### # Constants (LITERAL1) @@ -144,5 +147,37 @@ THERMOSTAT_MODE_OFF LITERAL1 THERMOSTAT_MODE_AUTO LITERAL1 THERMOSTAT_MODE_COOL LITERAL1 THERMOSTAT_MODE_HEAT LITERAL1 -THERMOSTAT_AUTO_MODE_DISABLED LITERAL1 -THERMOSTAT_AUTO_MODE_ENABLED LITERAL1 +MATTER_WIFI_CONNECTIVITY_CHANGE LITERAL1 +MATTER_THREAD_CONNECTIVITY_CHANGE LITERAL1 +MATTER_INTERNET_CONNECTIVITY_CHANGE LITERAL1 +MATTER_SERVICE_CONNECTIVITY_CHANGE LITERAL1 +MATTER_SERVICE_PROVISIONING_CHANGE LITERAL1 +MATTER_TIME_SYNC_CHANGE LITERAL1 +MATTER_CHIPOBLE_CONNECTION_ESTABLISHED LITERAL1 +MATTER_CHIPOBLE_CONNECTION_CLOSED LITERAL1 +MATTER_CLOSE_ALL_BLE_CONNECTIONS LITERAL1 +MATTER_WIFI_DEVICE_AVAILABLE LITERAL1 +MATTER_OPERATIONAL_NETWORK_STARTED LITERAL1 +MATTER_THREAD_STATE_CHANGE LITERAL1 +MATTER_THREAD_INTERFACE_STATE_CHANGE LITERAL1 +MATTER_CHIPOBLE_ADVERTISING_CHANGE LITERAL1 +MATTER_INTERFACE_IP_ADDRESS_CHANGED LITERAL1 +MATTER_COMMISSIONING_COMPLETE LITERAL1 +MATTER_FAIL_SAFE_TIMER_EXPIRED LITERAL1 +MATTER_OPERATIONAL_NETWORK_ENABLED LITERAL1 +MATTER_DNSSD_INITIALIZED LITERAL1 +MATTER_DNSSD_RESTART_NEEDED LITERAL1 +MATTER_BINDINGS_CHANGED_VIA_CLUSTER LITERAL1 +MATTER_OTA_STATE_CHANGED LITERAL1 +MATTER_SERVER_READY LITERAL1 +MATTER_BLE_DEINITIALIZED LITERAL1 +MATTER_ESP32_SPECIFIC_EVENT LITERAL1 +MATTER_COMMISSIONING_SESSION_STARTED LITERAL1 +MATTER_COMMISSIONING_SESSION_STOPPED LITERAL1 +MATTER_COMMISSIONING_WINDOW_OPEN LITERAL1 +MATTER_COMMISSIONING_WINDOW_CLOSED LITERAL1 +MATTER_FABRIC_WILL_BE_REMOVED LITERAL1 +MATTER_FABRIC_REMOVED LITERAL1 +MATTER_FABRIC_COMMITTED LITERAL1 +MATTER_FABRIC_UPDATED LITERAL1 +MATTER_ESP32_PUBLIC_SPECIFIC_EVENT LITERAL1 diff --git a/libraries/Matter/src/Matter.cpp b/libraries/Matter/src/Matter.cpp index af7c4c8657e..b16edfd85c1 100644 --- a/libraries/Matter/src/Matter.cpp +++ b/libraries/Matter/src/Matter.cpp @@ -28,7 +28,8 @@ constexpr auto k_timeout_seconds = 300; static bool _matter_has_started = false; static node::config_t node_config; -static node_t *deviceNode = NULL; +static node_t *deviceNode = nullptr; +ArduinoMatter::matterEventCB ArduinoMatter::_matterEventCB = nullptr; // This callback is called for every attribute update. The callback implementation shall // handle the desired attributes and return an appropriate error code. If the attribute @@ -42,7 +43,7 @@ static esp_err_t app_attribute_update_cb( switch (type) { case PRE_UPDATE: // Callback before updating the value in the database log_v("Attribute update callback: PRE_UPDATE"); - if (ep != NULL) { + if (ep != nullptr) { err = ep->attributeChangeCB(endpoint_id, cluster_id, attribute_id, val) ? ESP_OK : ESP_FAIL; } break; @@ -78,7 +79,7 @@ static esp_err_t app_identification_cb(identification::callback_type_t type, uin identifyIsActive = false; log_v("Identification callback: STOP"); } - if (ep != NULL) { + if (ep != nullptr) { err = ep->endpointIdentifyCB(endpoint_id, identifyIsActive) ? ESP_OK : ESP_FAIL; } @@ -89,21 +90,21 @@ static esp_err_t app_identification_cb(identification::callback_type_t type, uin static void app_event_cb(const ChipDeviceEvent *event, intptr_t arg) { switch (event->Type) { case chip::DeviceLayer::DeviceEventType::kInterfaceIpAddressChanged: - log_i( + log_d( "Interface %s Address changed", event->InterfaceIpAddressChanged.Type == chip::DeviceLayer::InterfaceIpChangeType::kIpV4_Assigned ? "IPv4" : "IPV6" ); break; - case chip::DeviceLayer::DeviceEventType::kCommissioningComplete: log_i("Commissioning complete"); break; - case chip::DeviceLayer::DeviceEventType::kFailSafeTimerExpired: log_i("Commissioning failed, fail safe timer expired"); break; - case chip::DeviceLayer::DeviceEventType::kCommissioningSessionStarted: log_i("Commissioning session started"); break; - case chip::DeviceLayer::DeviceEventType::kCommissioningSessionStopped: log_i("Commissioning session stopped"); break; - case chip::DeviceLayer::DeviceEventType::kCommissioningWindowOpened: log_i("Commissioning window opened"); break; - case chip::DeviceLayer::DeviceEventType::kCommissioningWindowClosed: log_i("Commissioning window closed"); break; + case chip::DeviceLayer::DeviceEventType::kCommissioningComplete: log_d("Commissioning complete"); break; + case chip::DeviceLayer::DeviceEventType::kFailSafeTimerExpired: log_d("Commissioning failed, fail safe timer expired"); break; + case chip::DeviceLayer::DeviceEventType::kCommissioningSessionStarted: log_d("Commissioning session started"); break; + case chip::DeviceLayer::DeviceEventType::kCommissioningSessionStopped: log_d("Commissioning session stopped"); break; + case chip::DeviceLayer::DeviceEventType::kCommissioningWindowOpened: log_d("Commissioning window opened"); break; + case chip::DeviceLayer::DeviceEventType::kCommissioningWindowClosed: log_d("Commissioning window closed"); break; case chip::DeviceLayer::DeviceEventType::kFabricRemoved: { - log_i("Fabric removed successfully"); + log_d("Fabric removed successfully"); if (chip::Server::GetInstance().GetFabricTable().FabricCount() == 0) { - log_i("No fabric left, opening commissioning window"); + log_d("No fabric left, opening commissioning window"); chip::CommissioningWindowManager &commissionMgr = chip::Server::GetInstance().GetCommissioningWindowManager(); constexpr auto kTimeoutSeconds = chip::System::Clock::Seconds16(k_timeout_seconds); if (!commissionMgr.IsCommissioningWindowOpen()) { @@ -116,12 +117,16 @@ static void app_event_cb(const ChipDeviceEvent *event, intptr_t arg) { } break; } - case chip::DeviceLayer::DeviceEventType::kFabricWillBeRemoved: log_i("Fabric will be removed"); break; - case chip::DeviceLayer::DeviceEventType::kFabricUpdated: log_i("Fabric is updated"); break; - case chip::DeviceLayer::DeviceEventType::kFabricCommitted: log_i("Fabric is committed"); break; - case chip::DeviceLayer::DeviceEventType::kBLEDeinitialized: log_i("BLE deinitialized and memory reclaimed"); break; + case chip::DeviceLayer::DeviceEventType::kFabricWillBeRemoved: log_d("Fabric will be removed"); break; + case chip::DeviceLayer::DeviceEventType::kFabricUpdated: log_d("Fabric is updated"); break; + case chip::DeviceLayer::DeviceEventType::kFabricCommitted: log_d("Fabric is committed"); break; + case chip::DeviceLayer::DeviceEventType::kBLEDeinitialized: log_d("BLE deinitialized and memory reclaimed"); break; default: break; } + // Check if the user-defined callback is set + if (ArduinoMatter::_matterEventCB != nullptr) { + ArduinoMatter::_matterEventCB(static_cast(event->Type), event); + } } void ArduinoMatter::_init() { diff --git a/libraries/Matter/src/Matter.h b/libraries/Matter/src/Matter.h index e54ceb47e5e..682a0498076 100644 --- a/libraries/Matter/src/Matter.h +++ b/libraries/Matter/src/Matter.h @@ -34,10 +34,136 @@ #include #include +// Matter Event types used when there is a user callback for Matter Events +enum matterEvent_t { + // Starting from 0x8000, these events are public and can be used by applications. + // Defined in CHIPDeviceEvent.h + + // WiFi Connectivity Change: Signals a change in connectivity of the device's WiFi station interface. + MATTER_WIFI_CONNECTIVITY_CHANGE = (uint16_t)chip::DeviceLayer::DeviceEventType::kWiFiConnectivityChange, + + // Thread Connectivity Change: Signals a change in connectivity of the device's Thread interface. + MATTER_THREAD_CONNECTIVITY_CHANGE = (uint16_t)chip::DeviceLayer::DeviceEventType::kThreadConnectivityChange, + + // Internet Connectivity Change: Signals a change in the device's ability to communicate via the Internet. + MATTER_INTERNET_CONNECTIVITY_CHANGE = (uint16_t)chip::DeviceLayer::DeviceEventType::kInternetConnectivityChange, + + // Service Connectivity Change: Signals a change in the device's ability to communicate with a chip-enabled service. + MATTER_SERVICE_CONNECTIVITY_CHANGE = (uint16_t)chip::DeviceLayer::DeviceEventType::kServiceConnectivityChange, + + // Service Provisioning Change: Signals a change to the device's service provisioning state. + MATTER_SERVICE_PROVISIONING_CHANGE = (uint16_t)chip::DeviceLayer::DeviceEventType::kServiceProvisioningChange, + + // Time Sync Change: Signals a change to the device's real time clock synchronization state. + MATTER_TIME_SYNC_CHANGE = (uint16_t)chip::DeviceLayer::DeviceEventType::kTimeSyncChange, + + // CHIPoBLE Connection Established: Signals that an external entity has established a new + // CHIPoBLE connection with the device. + MATTER_CHIPOBLE_CONNECTION_ESTABLISHED = (uint16_t)chip::DeviceLayer::DeviceEventType::kCHIPoBLEConnectionEstablished, + + // CHIPoBLE Connection Closed: Signals that an external entity has closed existing CHIPoBLE + // connection with the device. + MATTER_CHIPOBLE_CONNECTION_CLOSED = (uint16_t)chip::DeviceLayer::DeviceEventType::kCHIPoBLEConnectionClosed, + + // Request BLE connections to be closed. This is used in the supportsConcurrentConnection = False case. + MATTER_CLOSE_ALL_BLE_CONNECTIONS = (uint16_t)chip::DeviceLayer::DeviceEventType::kCloseAllBleConnections, + + // WiFi Device Available: When supportsConcurrentConnection = False, the ConnectNetwork + // command cannot start until the BLE device is closed and the Operation Network device (e.g. WiFi) has been started. + MATTER_WIFI_DEVICE_AVAILABLE = (uint16_t)chip::DeviceLayer::DeviceEventType::kWiFiDeviceAvailable, + + MATTER_OPERATIONAL_NETWORK_STARTED = (uint16_t)chip::DeviceLayer::DeviceEventType::kOperationalNetworkStarted, + + // Thread State Change: Signals that a state change has occurred in the Thread stack. + MATTER_THREAD_STATE_CHANGE = (uint16_t)chip::DeviceLayer::DeviceEventType::kThreadStateChange, + + // Thread Interface State Change: Signals that the state of the Thread network interface has changed. + MATTER_THREAD_INTERFACE_STATE_CHANGE = (uint16_t)chip::DeviceLayer::DeviceEventType::kThreadInterfaceStateChange, + + // CHIPoBLE Advertising Change: Signals that the state of CHIPoBLE advertising has changed. + MATTER_CHIPOBLE_ADVERTISING_CHANGE = (uint16_t)chip::DeviceLayer::DeviceEventType::kCHIPoBLEAdvertisingChange, + + // Interface IP Address Changed: IP address availability - either ipv4 or ipv6 + // addresses assigned to the underlying wifi/ethernet interface. + MATTER_INTERFACE_IP_ADDRESS_CHANGED = (uint16_t)chip::DeviceLayer::DeviceEventType::kInterfaceIpAddressChanged, + + // Commissioning Complete: Commissioning has completed by a call to the general + // commissioning cluster command. + MATTER_COMMISSIONING_COMPLETE = (uint16_t)chip::DeviceLayer::DeviceEventType::kCommissioningComplete, + + // Fail Safe Timer Expired: Signals that the fail-safe timer expired before + // the CommissioningComplete command was successfully invoked. + MATTER_FAIL_SAFE_TIMER_EXPIRED = (uint16_t)chip::DeviceLayer::DeviceEventType::kFailSafeTimerExpired, + + // Operational Network Enabled. + MATTER_OPERATIONAL_NETWORK_ENABLED = (uint16_t)chip::DeviceLayer::DeviceEventType::kOperationalNetworkEnabled, + + // DNS-SD Initialized: Signals that DNS-SD has been initialized and is ready to operate. + MATTER_DNSSD_INITIALIZED = (uint16_t)chip::DeviceLayer::DeviceEventType::kDnssdInitialized, + + // DNS-SD Restart Needed: Signals that DNS-SD backend was restarted and services must be published again. + MATTER_DNSSD_RESTART_NEEDED = (uint16_t)chip::DeviceLayer::DeviceEventType::kDnssdRestartNeeded, + + // Bindings Changed Via Cluster: Signals that bindings were updated. + MATTER_BINDINGS_CHANGED_VIA_CLUSTER = (uint16_t)chip::DeviceLayer::DeviceEventType::kBindingsChangedViaCluster, + + // OTA State Changed: Signals that the state of the OTA engine changed. + MATTER_OTA_STATE_CHANGED = (uint16_t)chip::DeviceLayer::DeviceEventType::kOtaStateChanged, + + // Server Ready: Server initialization has completed. Signals that all server components have been initialized + // and the node is ready to establish connections with other nodes. This event can be used to trigger on-boot actions + // that require sending messages to other nodes. + MATTER_SERVER_READY = (uint16_t)chip::DeviceLayer::DeviceEventType::kServerReady, + + // BLE Deinitialized: Signals that BLE stack is deinitialized and memory reclaimed + MATTER_BLE_DEINITIALIZED = (uint16_t)chip::DeviceLayer::DeviceEventType::kBLEDeinitialized, + + // Starting ESP32 Platform Specific Events from 0x9000 + MATTER_ESP32_SPECIFIC_EVENT, // value is previous + 1 + + // Commissioning Session Started: Signals that Commissioning session has started + MATTER_COMMISSIONING_SESSION_STARTED = (uint16_t)chip::DeviceLayer::DeviceEventType::kCommissioningSessionStarted, + + // Commissioning Session Stopped: Signals that Commissioning session has stopped + MATTER_COMMISSIONING_SESSION_STOPPED = (uint16_t)chip::DeviceLayer::DeviceEventType::kCommissioningSessionStopped, + + // Commissioning Window Opened: Signals that Commissioning window is now opened + MATTER_COMMISSIONING_WINDOW_OPEN = (uint16_t)chip::DeviceLayer::DeviceEventType::kCommissioningWindowOpened, + + // Commissioning Window Closed: Signals that Commissioning window is now closed + MATTER_COMMISSIONING_WINDOW_CLOSED = (uint16_t)chip::DeviceLayer::DeviceEventType::kCommissioningWindowClosed, + + // Fabric Will Be Removed: Signals that a fabric is about to be deleted. This allows actions to be taken that need the + // fabric to still be around before we delete it + MATTER_FABRIC_WILL_BE_REMOVED = (uint16_t)chip::DeviceLayer::DeviceEventType::kFabricWillBeRemoved, + + // Fabric Has Been Removed: Signals that a fabric is effectively deleted + MATTER_FABRIC_REMOVED = (uint16_t)chip::DeviceLayer::DeviceEventType::kFabricRemoved, + + // Fabric Has Been Committed: Signals that a fabric in Fabric Table is persisted to storage, by CommitPendingFabricData + MATTER_FABRIC_COMMITTED = (uint16_t)chip::DeviceLayer::DeviceEventType::kFabricCommitted, + + // Fabric Has Been Updated: Signals that operational credentials are changed, which may not be persistent. + // Can be used to affect what is needed for UpdateNOC prior to commit + MATTER_FABRIC_UPDATED = (uint16_t)chip::DeviceLayer::DeviceEventType::kFabricUpdated, + + // ESP32 Matter Events: These are custom ESP32 Matter events as defined in CHIPDevicePlatformEvent.h. + MATTER_ESP32_PUBLIC_SPECIFIC_EVENT = (uint16_t)chip::DeviceLayer::DeviceEventType::kRange_PublicPlatformSpecific, // ESPSystemEvent +}; + using namespace esp_matter; class ArduinoMatter { public: + // Matter Event Callback type + using matterEventCB = std::function; + // Matter Event Callback + static matterEventCB _matterEventCB; + // set the Matter Event Callback + static void onEvent(matterEventCB cb) { + _matterEventCB = cb; + } + static inline String getManualPairingCode() { // return the pairing code for manual pairing return String("34970112332"); diff --git a/libraries/Matter/src/MatterEndPoint.h b/libraries/Matter/src/MatterEndPoint.h index 5baa4747d18..95d3d3c08df 100644 --- a/libraries/Matter/src/MatterEndPoint.h +++ b/libraries/Matter/src/MatterEndPoint.h @@ -41,22 +41,22 @@ class MatterEndPoint { esp_matter::attribute_t *getAttribute(uint32_t cluster_id, uint32_t attribute_id) { if (endpoint_id == 0) { log_e("Endpoint ID is not set"); - return NULL; + return nullptr; } endpoint_t *endpoint = endpoint::get(node::get(), endpoint_id); - if (endpoint == NULL) { + if (endpoint == nullptr) { log_e("Endpoint [%d] not found", endpoint_id); - return NULL; + return nullptr; } cluster_t *cluster = cluster::get(endpoint, cluster_id); - if (cluster == NULL) { + if (cluster == nullptr) { log_e("Cluster [%d] not found", cluster_id); - return NULL; + return nullptr; } esp_matter::attribute_t *attribute = attribute::get(cluster, attribute_id); - if (attribute == NULL) { + if (attribute == nullptr) { log_e("Attribute [%d] not found", attribute_id); - return NULL; + return nullptr; } return attribute; } @@ -64,7 +64,7 @@ class MatterEndPoint { // get the value of an attribute from its cluster id and attribute it bool getAttributeVal(uint32_t cluster_id, uint32_t attribute_id, esp_matter_attr_val_t *attrVal) { esp_matter::attribute_t *attribute = getAttribute(cluster_id, attribute_id); - if (attribute == NULL) { + if (attribute == nullptr) { return false; } if (attribute::get_val(attribute, attrVal) == ESP_OK) { @@ -78,7 +78,7 @@ class MatterEndPoint { // set the value of an attribute from its cluster id and attribute it bool setAttributeVal(uint32_t cluster_id, uint32_t attribute_id, esp_matter_attr_val_t *attrVal) { esp_matter::attribute_t *attribute = getAttribute(cluster_id, attribute_id); - if (attribute == NULL) { + if (attribute == nullptr) { return false; } if (attribute::set_val(attribute, attrVal) == ESP_OK) { @@ -117,6 +117,6 @@ class MatterEndPoint { protected: uint16_t endpoint_id = 0; - EndPointIdentifyCB _onEndPointIdentifyCB = NULL; + EndPointIdentifyCB _onEndPointIdentifyCB = nullptr; }; #endif /* CONFIG_ESP_MATTER_ENABLE_DATA_MODEL */ From 01e4d5debbbc1b35dd0dfd6853a29b8ebba02f10 Mon Sep 17 00:00:00 2001 From: Sugar Glider Date: Sat, 21 Jun 2025 09:45:26 -0300 Subject: [PATCH 060/173] feat(uart): fixes loopback function after IDF changes (#11492) * feat(uart): fixes loopback function after IDF changes IDF 5.4.1 has added a new function called uart_release_pin() that is called whenever new pins are set or when uart driver is deleted. This has a side effect that causes RX pin to never work again with the loopback function. Other changes also have removed some GPIO setup that was necessary for the GPIO loopback mode work. The PR forces a full RX Pin setup in order to make it work in GPIO Matrix with Loopback TX Signal * feat(uart): adds missing include file * feat(uart): removes not necessary part of the code * fix(uart): commentaries style fix Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> * ci(pre-commit): Apply automatic fixes --------- Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> Co-authored-by: pre-commit-ci-lite[bot] <117423508+pre-commit-ci-lite[bot]@users.noreply.github.com> --- cores/esp32/esp32-hal-uart.c | 46 +++++++++--------------------------- 1 file changed, 11 insertions(+), 35 deletions(-) diff --git a/cores/esp32/esp32-hal-uart.c b/cores/esp32/esp32-hal-uart.c index 5311aff4f37..3c9e6bf178b 100644 --- a/cores/esp32/esp32-hal-uart.c +++ b/cores/esp32/esp32-hal-uart.c @@ -32,10 +32,11 @@ #include "driver/gpio.h" #include "hal/gpio_hal.h" #include "esp_rom_gpio.h" +#include "esp_private/gpio.h" #include "driver/rtc_io.h" #include "driver/lp_io.h" -#include "soc/uart_periph.h" +#include "soc/uart_pins.h" #include "esp_private/uart_share_hw_ctrl.h" static int s_uart_debug_nr = 0; // UART number for debug output @@ -1383,39 +1384,9 @@ unsigned long uartDetectBaudrate(uart_t *uart) { } /* - These functions are for testing purpose only and can be used in Arduino Sketches - Those are used in the UART examples -*/ - -/* - This is intended to make an internal loopback connection using IOMUX - The function uart_internal_loopback() shall be used right after Arduino Serial.begin(...) - This code "replaces" the physical wiring for connecting TX <--> RX in a loopback -*/ - -// gets the right TX or RX SIGNAL, based on the UART number from gpio_sig_map.h -#ifdef CONFIG_IDF_TARGET_ESP32P4 -#define UART_TX_SIGNAL(uartNumber) \ - (uartNumber == UART_NUM_0 \ - ? UART0_TXD_PAD_OUT_IDX \ - : (uartNumber == UART_NUM_1 \ - ? UART1_TXD_PAD_OUT_IDX \ - : (uartNumber == UART_NUM_2 ? UART2_TXD_PAD_OUT_IDX : (uartNumber == UART_NUM_3 ? UART3_TXD_PAD_OUT_IDX : UART4_TXD_PAD_OUT_IDX)))) -#define UART_RX_SIGNAL(uartNumber) \ - (uartNumber == UART_NUM_0 \ - ? UART0_RXD_PAD_IN_IDX \ - : (uartNumber == UART_NUM_1 \ - ? UART1_RXD_PAD_IN_IDX \ - : (uartNumber == UART_NUM_2 ? UART2_RXD_PAD_IN_IDX : (uartNumber == UART_NUM_3 ? UART3_RXD_PAD_IN_IDX : UART4_RXD_PAD_IN_IDX)))) -#else -#if SOC_UART_HP_NUM > 2 -#define UART_TX_SIGNAL(uartNumber) (uartNumber == UART_NUM_0 ? U0TXD_OUT_IDX : (uartNumber == UART_NUM_1 ? U1TXD_OUT_IDX : U2TXD_OUT_IDX)) -#define UART_RX_SIGNAL(uartNumber) (uartNumber == UART_NUM_0 ? U0RXD_IN_IDX : (uartNumber == UART_NUM_1 ? U1RXD_IN_IDX : U2RXD_IN_IDX)) -#else -#define UART_TX_SIGNAL(uartNumber) (uartNumber == UART_NUM_0 ? U0TXD_OUT_IDX : U1TXD_OUT_IDX) -#define UART_RX_SIGNAL(uartNumber) (uartNumber == UART_NUM_0 ? U0RXD_IN_IDX : U1RXD_IN_IDX) -#endif -#endif // ifdef CONFIG_IDF_TARGET_ESP32P4 + * These functions are for testing purposes only and can be used in Arduino Sketches. + * They are utilized in the UART examples and CI. + */ /* This function internally binds defined UARTs TX signal with defined RX pin of any UART (same or different). @@ -1427,7 +1398,12 @@ void uart_internal_loopback(uint8_t uartNum, int8_t rxPin) { log_e("UART%d is not supported for loopback or RX pin %d is invalid.", uartNum, rxPin); return; } - esp_rom_gpio_connect_out_signal(rxPin, UART_TX_SIGNAL(uartNum), false, false); + // forces rxPin to use GPIO Matrix and setup the pin to receive UART TX Signal - IDF 5.4.1 Change with uart_release_pin() + gpio_func_sel((gpio_num_t)rxPin, PIN_FUNC_GPIO); + gpio_pullup_en((gpio_num_t)rxPin); + gpio_input_enable((gpio_num_t)rxPin); + esp_rom_gpio_connect_in_signal(rxPin, uart_periph_signal[uartNum].pins[SOC_UART_RX_PIN_IDX].signal, false); + esp_rom_gpio_connect_out_signal(rxPin, uart_periph_signal[uartNum].pins[SOC_UART_TX_PIN_IDX].signal, false, false); } /* From 9d84c78cf2d44911639530e54a9dbb9ee9164f6c Mon Sep 17 00:00:00 2001 From: Sugar Glider Date: Sun, 22 Jun 2025 15:02:36 -0300 Subject: [PATCH 061/173] feat(uart): fixes pin attach for any IDF 5.x (#11499) * feat(uart): fixes pin attach for any IDF 5.x * fix(uart): commentary typo error Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> * fix(uart): commentary typo error Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> * fix(uart): commentary typo error Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> * fix(uart): commentary style fix * ci(pre-commit): Apply automatic fixes --------- Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> Co-authored-by: pre-commit-ci-lite[bot] <117423508+pre-commit-ci-lite[bot]@users.noreply.github.com> --- cores/esp32/esp32-hal-uart.c | 131 +++++++++++++++++++++++++++++++++-- 1 file changed, 126 insertions(+), 5 deletions(-) diff --git a/cores/esp32/esp32-hal-uart.c b/cores/esp32/esp32-hal-uart.c index 3c9e6bf178b..2163e9b5f42 100644 --- a/cores/esp32/esp32-hal-uart.c +++ b/cores/esp32/esp32-hal-uart.c @@ -295,6 +295,125 @@ static bool _uartDetachBus_RTS(void *busptr) { return _uartDetachPins(bus->num, UART_PIN_NO_CHANGE, UART_PIN_NO_CHANGE, UART_PIN_NO_CHANGE, bus->_rtsPin); } +static bool _uartTrySetIomuxPin(uart_port_t uart_num, int io_num, uint32_t idx) { + // Store a pointer to the default pin, to optimize access to its fields. + const uart_periph_sig_t *upin = &uart_periph_signal[uart_num].pins[idx]; + + // In theory, if default_gpio is -1, iomux_func should also be -1, but let's be safe and test both. + if (upin->iomux_func == -1 || upin->default_gpio == -1 || upin->default_gpio != io_num) { + return false; + } + + // Assign the correct function to the GPIO. + assert(upin->iomux_func != -1); + if (uart_num < SOC_UART_HP_NUM) { + gpio_iomux_out(io_num, upin->iomux_func, false); + // If the pin is input, we also have to redirect the signal, in order to bypass the GPIO matrix. + if (upin->input) { + gpio_iomux_in(io_num, upin->signal); + } + } +#if (SOC_UART_LP_NUM >= 1) && (SOC_RTCIO_PIN_COUNT >= 1) + else { + if (upin->input) { + rtc_gpio_set_direction(io_num, RTC_GPIO_MODE_INPUT_ONLY); + } else { + rtc_gpio_set_direction(io_num, RTC_GPIO_MODE_OUTPUT_ONLY); + } + rtc_gpio_init(io_num); + rtc_gpio_iomux_func_sel(io_num, upin->iomux_func); + } +#endif + return true; +} + +static esp_err_t _uartInternalSetPin(uart_port_t uart_num, int tx_io_num, int rx_io_num, int rts_io_num, int cts_io_num) { + // Since an IO cannot route peripheral signals via IOMUX and GPIO matrix at the same time, + // if tx and rx share the same IO, both signals need to be routed to IOs through GPIO matrix + bool tx_rx_same_io = (tx_io_num == rx_io_num); + + // In the following statements, if the io_num is negative, no need to configure anything. + if (tx_io_num >= 0) { +#if CONFIG_ESP_SLEEP_GPIO_RESET_WORKAROUND || CONFIG_PM_SLP_DISABLE_GPIO + // In such case, IOs are going to switch to sleep configuration (isolate) when entering sleep for power saving reason + // But TX IO in isolate state could write garbled data to the other end + // Therefore, we should disable the switch of the TX pin to sleep configuration + gpio_sleep_sel_dis(tx_io_num); +#endif + if (tx_rx_same_io || !_uartTrySetIomuxPin(uart_num, tx_io_num, SOC_UART_TX_PIN_IDX)) { + if (uart_num < SOC_UART_HP_NUM) { + gpio_func_sel(tx_io_num, PIN_FUNC_GPIO); + esp_rom_gpio_connect_out_signal(tx_io_num, UART_PERIPH_SIGNAL(uart_num, SOC_UART_TX_PIN_IDX), 0, 0); + // output enable is set inside esp_rom_gpio_connect_out_signal func after the signal is connected + // (output enabled too early may cause unnecessary level change at the pad) + } +#if SOC_LP_GPIO_MATRIX_SUPPORTED + else { + rtc_gpio_init(tx_io_num); // set as a LP_GPIO pin + lp_gpio_connect_out_signal(tx_io_num, UART_PERIPH_SIGNAL(uart_num, SOC_UART_TX_PIN_IDX), 0, 0); + // output enable is set inside lp_gpio_connect_out_signal func after the signal is connected + } +#endif + } + } + + if (rx_io_num >= 0) { +#if CONFIG_ESP_SLEEP_GPIO_RESET_WORKAROUND || CONFIG_PM_SLP_DISABLE_GPIO + // In such case, IOs are going to switch to sleep configuration (isolate) when entering sleep for power saving reason + // But RX IO in isolate state could receive garbled data into FIFO, which is not desired + // Therefore, we should disable the switch of the RX pin to sleep configuration + gpio_sleep_sel_dis(rx_io_num); +#endif + if (tx_rx_same_io || !_uartTrySetIomuxPin(uart_num, rx_io_num, SOC_UART_RX_PIN_IDX)) { + if (uart_num < SOC_UART_HP_NUM) { + gpio_input_enable(rx_io_num); + esp_rom_gpio_connect_in_signal(rx_io_num, UART_PERIPH_SIGNAL(uart_num, SOC_UART_RX_PIN_IDX), 0); + } +#if SOC_LP_GPIO_MATRIX_SUPPORTED + else { + rtc_gpio_mode_t mode = (tx_rx_same_io ? RTC_GPIO_MODE_INPUT_OUTPUT : RTC_GPIO_MODE_INPUT_ONLY); + rtc_gpio_set_direction(rx_io_num, mode); + if (!tx_rx_same_io) { // set the same pin again as a LP_GPIO will overwrite connected out_signal, not desired, so skip + rtc_gpio_init(rx_io_num); // set as a LP_GPIO pin + } + lp_gpio_connect_in_signal(rx_io_num, UART_PERIPH_SIGNAL(uart_num, SOC_UART_RX_PIN_IDX), 0); + } +#endif + } + } + + if (rts_io_num >= 0 && !_uartTrySetIomuxPin(uart_num, rts_io_num, SOC_UART_RTS_PIN_IDX)) { + if (uart_num < SOC_UART_HP_NUM) { + gpio_func_sel(rts_io_num, PIN_FUNC_GPIO); + esp_rom_gpio_connect_out_signal(rts_io_num, UART_PERIPH_SIGNAL(uart_num, SOC_UART_RTS_PIN_IDX), 0, 0); + // output enable is set inside esp_rom_gpio_connect_out_signal func after the signal is connected + } +#if SOC_LP_GPIO_MATRIX_SUPPORTED + else { + rtc_gpio_init(rts_io_num); // set as a LP_GPIO pin + lp_gpio_connect_out_signal(rts_io_num, UART_PERIPH_SIGNAL(uart_num, SOC_UART_RTS_PIN_IDX), 0, 0); + // output enable is set inside lp_gpio_connect_out_signal func after the signal is connected + } +#endif + } + + if (cts_io_num >= 0 && !_uartTrySetIomuxPin(uart_num, cts_io_num, SOC_UART_CTS_PIN_IDX)) { + if (uart_num < SOC_UART_HP_NUM) { + gpio_pullup_en(cts_io_num); + gpio_input_enable(cts_io_num); + esp_rom_gpio_connect_in_signal(cts_io_num, UART_PERIPH_SIGNAL(uart_num, SOC_UART_CTS_PIN_IDX), 0); + } +#if SOC_LP_GPIO_MATRIX_SUPPORTED + else { + rtc_gpio_set_direction(cts_io_num, RTC_GPIO_MODE_INPUT_ONLY); + rtc_gpio_init(cts_io_num); // set as a LP_GPIO pin + lp_gpio_connect_in_signal(cts_io_num, UART_PERIPH_SIGNAL(uart_num, SOC_UART_CTS_PIN_IDX), 0); + } +#endif + } + return ESP_OK; +} + // Attach function for UART // connects the IO Pad, set Paripheral Manager and internal UART structure data static bool _uartAttachPins(uint8_t uart_num, int8_t rxPin, int8_t txPin, int8_t ctsPin, int8_t rtsPin) { @@ -307,7 +426,7 @@ static bool _uartAttachPins(uint8_t uart_num, int8_t rxPin, int8_t txPin, int8_t //log_v("attaching UART%d pins: prev,new RX(%d,%d) TX(%d,%d) CTS(%d,%d) RTS(%d,%d)", uart_num, // uart->_rxPin, rxPin, uart->_txPin, txPin, uart->_ctsPin, ctsPin, uart->_rtsPin, rtsPin); vTaskDelay(10); - // IDF uart_set_pin() checks if the pin is used within LP UART and if it is a valid RTC IO pin + // IDF _uartInternalSetPin() checks if the pin is used within LP UART and if it is a valid RTC IO pin // No need for Arduino Layer to check it again bool retCode = true; if (rxPin >= 0) { @@ -316,7 +435,7 @@ static bool _uartAttachPins(uint8_t uart_num, int8_t rxPin, int8_t txPin, int8_t perimanClearPinBus(rxPin); } // connect RX Pad - bool ret = ESP_OK == uart_set_pin(uart->num, UART_PIN_NO_CHANGE, rxPin, UART_PIN_NO_CHANGE, UART_PIN_NO_CHANGE); + bool ret = ESP_OK == _uartInternalSetPin(uart->num, UART_PIN_NO_CHANGE, rxPin, UART_PIN_NO_CHANGE, UART_PIN_NO_CHANGE); #if SOC_UART_LP_NUM >= 1 if (ret && uart_num >= SOC_UART_HP_NUM) { // it is a LP UART NUM ret &= lp_uart_config_io(uart->num, rxPin, RTC_GPIO_MODE_INPUT_ONLY, SOC_UART_RX_PIN_IDX); @@ -339,7 +458,7 @@ static bool _uartAttachPins(uint8_t uart_num, int8_t rxPin, int8_t txPin, int8_t perimanClearPinBus(txPin); } // connect TX Pad - bool ret = ESP_OK == uart_set_pin(uart->num, txPin, UART_PIN_NO_CHANGE, UART_PIN_NO_CHANGE, UART_PIN_NO_CHANGE); + bool ret = ESP_OK == _uartInternalSetPin(uart->num, txPin, UART_PIN_NO_CHANGE, UART_PIN_NO_CHANGE, UART_PIN_NO_CHANGE); #if SOC_UART_LP_NUM >= 1 if (ret && uart_num >= SOC_UART_HP_NUM) { // it is a LP UART NUM ret &= lp_uart_config_io(uart->num, txPin, RTC_GPIO_MODE_OUTPUT_ONLY, SOC_UART_TX_PIN_IDX); @@ -362,7 +481,7 @@ static bool _uartAttachPins(uint8_t uart_num, int8_t rxPin, int8_t txPin, int8_t perimanClearPinBus(ctsPin); } // connect CTS Pad - bool ret = ESP_OK == uart_set_pin(uart->num, UART_PIN_NO_CHANGE, UART_PIN_NO_CHANGE, UART_PIN_NO_CHANGE, ctsPin); + bool ret = ESP_OK == _uartInternalSetPin(uart->num, UART_PIN_NO_CHANGE, UART_PIN_NO_CHANGE, UART_PIN_NO_CHANGE, ctsPin); #if SOC_UART_LP_NUM >= 1 if (ret && uart_num >= SOC_UART_HP_NUM) { // it is a LP UART NUM ret &= lp_uart_config_io(uart->num, ctsPin, RTC_GPIO_MODE_INPUT_ONLY, SOC_UART_CTS_PIN_IDX); @@ -385,7 +504,7 @@ static bool _uartAttachPins(uint8_t uart_num, int8_t rxPin, int8_t txPin, int8_t perimanClearPinBus(rtsPin); } // connect RTS Pad - bool ret = ESP_OK == uart_set_pin(uart->num, UART_PIN_NO_CHANGE, UART_PIN_NO_CHANGE, rtsPin, UART_PIN_NO_CHANGE); + bool ret = ESP_OK == _uartInternalSetPin(uart->num, UART_PIN_NO_CHANGE, UART_PIN_NO_CHANGE, rtsPin, UART_PIN_NO_CHANGE); #if SOC_UART_LP_NUM >= 1 if (ret && uart_num >= SOC_UART_HP_NUM) { // it is a LP UART NUM ret &= lp_uart_config_io(uart->num, rtsPin, RTC_GPIO_MODE_OUTPUT_ONLY, SOC_UART_RTS_PIN_IDX); @@ -1398,11 +1517,13 @@ void uart_internal_loopback(uint8_t uartNum, int8_t rxPin) { log_e("UART%d is not supported for loopback or RX pin %d is invalid.", uartNum, rxPin); return; } +#if 0 // leave this code here for future reference and need // forces rxPin to use GPIO Matrix and setup the pin to receive UART TX Signal - IDF 5.4.1 Change with uart_release_pin() gpio_func_sel((gpio_num_t)rxPin, PIN_FUNC_GPIO); gpio_pullup_en((gpio_num_t)rxPin); gpio_input_enable((gpio_num_t)rxPin); esp_rom_gpio_connect_in_signal(rxPin, uart_periph_signal[uartNum].pins[SOC_UART_RX_PIN_IDX].signal, false); +#endif esp_rom_gpio_connect_out_signal(rxPin, uart_periph_signal[uartNum].pins[SOC_UART_TX_PIN_IDX].signal, false, false); } From febca6b2f1d02b9252c2f5edc2aa312129c68720 Mon Sep 17 00:00:00 2001 From: me-no-dev Date: Mon, 23 Jun 2025 12:11:32 +0300 Subject: [PATCH 062/173] fix(p4): Update hosted and wifi_remote components --- idf_component.yml | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/idf_component.yml b/idf_component.yml index 257890257cc..82f14ea554a 100644 --- a/idf_component.yml +++ b/idf_component.yml @@ -107,11 +107,11 @@ dependencies: rules: - if: "target in [esp32s3]" espressif/esp_hosted: - version: "^2.0.0" + version: "^2.0.12" rules: - if: "target == esp32p4" espressif/esp_wifi_remote: - version: "~0.9.2" + version: "^0.13.0" rules: - if: "target == esp32p4" espressif/libsodium: From 1c79eb823cf4beb8465b987bacea1347574979db Mon Sep 17 00:00:00 2001 From: Lucas Saavedra Vaz <32426024+lucasssvaz@users.noreply.github.com> Date: Mon, 23 Jun 2025 08:47:49 -0300 Subject: [PATCH 063/173] feat(NimBLE): Add support for NimBLE (#11424) * feat(NimBLE): Add support for NimBLE Co-authored-by: h2zero * ci(pre-commit): Apply automatic fixes * fix(nimble): Fix typo in BLEClient --------- Co-authored-by: h2zero Co-authored-by: pre-commit-ci-lite[bot] <117423508+pre-commit-ci-lite[bot]@users.noreply.github.com> --- cores/esp32/esp32-hal-bt.c | 6 +- cores/esp32/esp32-hal-misc.c | 11 +- libraries/BLE/README.md | 8 +- .../BLE5_extended_scan/BLE5_extended_scan.ino | 6 +- .../BLE/examples/BLE5_extended_scan/ci.json | 3 +- .../BLE5_multi_advertising.ino | 4 +- .../examples/BLE5_multi_advertising/ci.json | 3 +- .../BLE5_periodic_advertising.ino | 4 +- .../BLE5_periodic_advertising/ci.json | 3 +- .../BLE5_periodic_sync/BLE5_periodic_sync.ino | 6 +- .../BLE/examples/BLE5_periodic_sync/ci.json | 3 +- .../Beacon_Scanner/Beacon_Scanner.ino | 42 +- libraries/BLE/examples/Client/Client.ino | 2 + libraries/BLE/examples/Notify/Notify.ino | 1 + .../Server_multiconnect.ino | 52 +- libraries/BLE/examples/UART/UART.ino | 13 +- libraries/BLE/library.properties | 2 +- libraries/BLE/src/BLE2901.cpp | 47 +- libraries/BLE/src/BLE2901.h | 39 +- libraries/BLE/src/BLE2902.cpp | 73 +- libraries/BLE/src/BLE2902.h | 46 +- libraries/BLE/src/BLE2904.cpp | 36 +- libraries/BLE/src/BLE2904.h | 39 +- libraries/BLE/src/BLEAddress.cpp | 184 ++- libraries/BLE/src/BLEAddress.h | 85 +- libraries/BLE/src/BLEAdvertisedDevice.cpp | 165 ++- libraries/BLE/src/BLEAdvertisedDevice.h | 154 ++- libraries/BLE/src/BLEAdvertising.cpp | 915 +++++++++++-- libraries/BLE/src/BLEAdvertising.h | 210 ++- libraries/BLE/src/BLEBeacon.cpp | 47 +- libraries/BLE/src/BLEBeacon.h | 20 + libraries/BLE/src/BLECharacteristic.cpp | 1198 +++++++++++------ libraries/BLE/src/BLECharacteristic.h | 261 +++- libraries/BLE/src/BLECharacteristicMap.cpp | 76 +- libraries/BLE/src/BLEClient.cpp | 1009 +++++++++++--- libraries/BLE/src/BLEClient.h | 191 ++- libraries/BLE/src/BLEConnInfo.h | 110 ++ libraries/BLE/src/BLEDescriptor.cpp | 297 ++-- libraries/BLE/src/BLEDescriptor.h | 127 +- libraries/BLE/src/BLEDescriptorMap.cpp | 102 +- libraries/BLE/src/BLEDevice.cpp | 869 ++++++++---- libraries/BLE/src/BLEDevice.h | 246 +++- libraries/BLE/src/BLEEddystoneTLM.cpp | 8 +- .../BLE/src/BLEEddystoneTLM.cppwithheadder | 202 --- libraries/BLE/src/BLEEddystoneTLM.h | 10 +- libraries/BLE/src/BLEEddystoneURL.cpp | 12 +- libraries/BLE/src/BLEEddystoneURL.h | 8 + libraries/BLE/src/BLEEddystoneURL.h.orig | 66 - libraries/BLE/src/BLEExceptions.cpp | 5 + libraries/BLE/src/BLEHIDDevice.cpp | 39 +- libraries/BLE/src/BLEHIDDevice.h | 8 +- libraries/BLE/src/BLERemoteCharacteristic.cpp | 975 ++++++++++---- libraries/BLE/src/BLERemoteCharacteristic.h | 140 +- libraries/BLE/src/BLERemoteDescriptor.cpp | 393 +++++- libraries/BLE/src/BLERemoteDescriptor.h | 76 +- libraries/BLE/src/BLERemoteService.cpp | 369 +++-- libraries/BLE/src/BLERemoteService.h | 87 +- libraries/BLE/src/BLEScan.cpp | 690 +++++++--- libraries/BLE/src/BLEScan.h | 148 +- libraries/BLE/src/BLESecurity.cpp | 140 +- libraries/BLE/src/BLESecurity.h | 144 +- libraries/BLE/src/BLEServer.cpp | 748 ++++++++-- libraries/BLE/src/BLEServer.h | 217 ++- libraries/BLE/src/BLEService.cpp | 429 ++++-- libraries/BLE/src/BLEService.h | 126 +- libraries/BLE/src/BLEServiceMap.cpp | 40 +- libraries/BLE/src/BLEUUID.cpp | 375 +++--- libraries/BLE/src/BLEUUID.h | 101 +- libraries/BLE/src/BLEUtils.cpp | 682 ++++++++-- libraries/BLE/src/BLEUtils.h | 113 +- libraries/BLE/src/BLEValue.cpp | 18 +- libraries/BLE/src/BLEValue.h | 25 +- 72 files changed, 9898 insertions(+), 3211 deletions(-) create mode 100644 libraries/BLE/src/BLEConnInfo.h delete mode 100644 libraries/BLE/src/BLEEddystoneTLM.cppwithheadder delete mode 100644 libraries/BLE/src/BLEEddystoneURL.h.orig diff --git a/cores/esp32/esp32-hal-bt.c b/cores/esp32/esp32-hal-bt.c index 5d512d448a3..a2a603bf5b9 100644 --- a/cores/esp32/esp32-hal-bt.c +++ b/cores/esp32/esp32-hal-bt.c @@ -15,7 +15,7 @@ #include "esp32-hal-bt.h" #if SOC_BT_SUPPORTED -#if defined(CONFIG_BT_BLUEDROID_ENABLED) && __has_include("esp_bt.h") +#if (defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED)) && __has_include("esp_bt.h") #if CONFIG_IDF_TARGET_ESP32 bool btInUse() { @@ -116,7 +116,7 @@ bool btStop() { return false; } -#else // CONFIG_BT_ENABLED +#else // !__has_include("esp_bt.h") || !(defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED)) bool btStarted() { return false; } @@ -129,6 +129,6 @@ bool btStop() { return false; } -#endif /* CONFIG_BT_ENABLED */ +#endif /* !__has_include("esp_bt.h") || !(defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED)) */ #endif /* SOC_BT_SUPPORTED */ diff --git a/cores/esp32/esp32-hal-misc.c b/cores/esp32/esp32-hal-misc.c index 9c1eaf6ba25..4c8a4e5beab 100644 --- a/cores/esp32/esp32-hal-misc.c +++ b/cores/esp32/esp32-hal-misc.c @@ -25,12 +25,13 @@ #include "esp_ota_ops.h" #endif //CONFIG_APP_ROLLBACK_ENABLE #include "esp_private/startup_internal.h" -#if defined(CONFIG_BT_BLUEDROID_ENABLED) && SOC_BT_SUPPORTED && __has_include("esp_bt.h") +#if (defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED)) && SOC_BT_SUPPORTED && __has_include("esp_bt.h") #include "esp_bt.h" -#endif //CONFIG_BT_BLUEDROID_ENABLED +#endif #include #include "soc/rtc.h" -#if !defined(CONFIG_IDF_TARGET_ESP32C2) && !defined(CONFIG_IDF_TARGET_ESP32C6) && !defined(CONFIG_IDF_TARGET_ESP32H2) && !defined(CONFIG_IDF_TARGET_ESP32P4) && !defined(CONFIG_IDF_TARGET_ESP32C5) +#if !defined(CONFIG_IDF_TARGET_ESP32C2) && !defined(CONFIG_IDF_TARGET_ESP32C6) && !defined(CONFIG_IDF_TARGET_ESP32H2) && !defined(CONFIG_IDF_TARGET_ESP32P4) \ + && !defined(CONFIG_IDF_TARGET_ESP32C5) #include "soc/rtc_cntl_reg.h" #include "soc/syscon_reg.h" #endif @@ -245,7 +246,7 @@ bool verifyRollbackLater() { } #endif -#ifdef CONFIG_BT_BLUEDROID_ENABLED +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) #if CONFIG_IDF_TARGET_ESP32 //overwritten in esp32-hal-bt.c bool btInUse() __attribute__((weak)); @@ -307,7 +308,7 @@ void initArduino() { if (err) { log_e("Failed to initialize NVS! Error: %u", err); } -#if defined(CONFIG_BT_BLUEDROID_ENABLED) && SOC_BT_SUPPORTED +#if (defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED)) && SOC_BT_SUPPORTED if (!btInUse()) { esp_bt_controller_mem_release(ESP_BT_MODE_BTDM); } diff --git a/libraries/BLE/README.md b/libraries/BLE/README.md index eb70ee9ff00..759c8526f0f 100644 --- a/libraries/BLE/README.md +++ b/libraries/BLE/README.md @@ -1,8 +1,12 @@ # ESP32 BLE for Arduino The Arduino IDE provides an excellent library package manager where versions of libraries can be downloaded and installed. This Github project provides the repository for the ESP32 BLE support for Arduino. -The original source of the project, **which is not maintained anymore**, can be found here: https://github.com/nkolban/esp32-snippets +The original source of the Bluedroid project, **which is not maintained anymore**, can be found here: https://github.com/nkolban/esp32-snippets -Issues and questions should be raised here: https://github.com/espressif/arduino-esp32/issues
(please don't use https://github.com/nkolban/esp32-snippets/issues!) +Some parts of the NimBLE implementation are based on the work of h2zero, which can be found here: https://github.com/h2zero/NimBLE-Arduino + +Issues and questions should be raised here: https://github.com/espressif/arduino-esp32/issues
(please don't use https://github.com/nkolban/esp32-snippets/issues or https://github.com/h2zero/NimBLE-Arduino/issues!) Documentation for using the library can be found here: https://github.com/nkolban/esp32-snippets/tree/master/Documentation + +For a more customizable and feature-rich implementation of the NimBLE stack, you can use the [NimBLE-Arduino](https://github.com/h2zero/NimBLE-Arduino) library. diff --git a/libraries/BLE/examples/BLE5_extended_scan/BLE5_extended_scan.ino b/libraries/BLE/examples/BLE5_extended_scan/BLE5_extended_scan.ino index 42daff86835..e50d7339df6 100644 --- a/libraries/BLE/examples/BLE5_extended_scan/BLE5_extended_scan.ino +++ b/libraries/BLE/examples/BLE5_extended_scan/BLE5_extended_scan.ino @@ -7,8 +7,10 @@ author: chegewara */ -#ifndef SOC_BLE_50_SUPPORTED -#warning "This SoC does not support BLE5. Try using ESP32-C3, or ESP32-S3" +#ifndef CONFIG_BLUEDROID_ENABLED +#error "NimBLE does not support extended scan yet. Try using Bluedroid." +#elif !defined(SOC_BLE_50_SUPPORTED) +#error "This SoC does not support BLE5. Try using ESP32-C3, or ESP32-S3" #else #include diff --git a/libraries/BLE/examples/BLE5_extended_scan/ci.json b/libraries/BLE/examples/BLE5_extended_scan/ci.json index 184cc25a2b0..8e938055cf2 100644 --- a/libraries/BLE/examples/BLE5_extended_scan/ci.json +++ b/libraries/BLE/examples/BLE5_extended_scan/ci.json @@ -1,6 +1,7 @@ { "fqbn_append": "PartitionScheme=huge_app", "requires": [ - "CONFIG_SOC_BLE_50_SUPPORTED=y" + "CONFIG_SOC_BLE_50_SUPPORTED=y", + "CONFIG_BLUEDROID_ENABLED=y" ] } diff --git a/libraries/BLE/examples/BLE5_multi_advertising/BLE5_multi_advertising.ino b/libraries/BLE/examples/BLE5_multi_advertising/BLE5_multi_advertising.ino index c4d614786b0..ee25f8e95ef 100644 --- a/libraries/BLE/examples/BLE5_multi_advertising/BLE5_multi_advertising.ino +++ b/libraries/BLE/examples/BLE5_multi_advertising/BLE5_multi_advertising.ino @@ -6,7 +6,9 @@ author: chegewara */ -#ifndef CONFIG_BT_BLE_50_FEATURES_SUPPORTED +#ifndef CONFIG_BLUEDROID_ENABLED +#error "NimBLE does not support multi advertising yet. Try using Bluedroid." +#elif !defined(CONFIG_BT_BLE_50_FEATURES_SUPPORTED) #error "This SoC does not support BLE5. Try using ESP32-C3, or ESP32-S3" #else diff --git a/libraries/BLE/examples/BLE5_multi_advertising/ci.json b/libraries/BLE/examples/BLE5_multi_advertising/ci.json index 184cc25a2b0..8e938055cf2 100644 --- a/libraries/BLE/examples/BLE5_multi_advertising/ci.json +++ b/libraries/BLE/examples/BLE5_multi_advertising/ci.json @@ -1,6 +1,7 @@ { "fqbn_append": "PartitionScheme=huge_app", "requires": [ - "CONFIG_SOC_BLE_50_SUPPORTED=y" + "CONFIG_SOC_BLE_50_SUPPORTED=y", + "CONFIG_BLUEDROID_ENABLED=y" ] } diff --git a/libraries/BLE/examples/BLE5_periodic_advertising/BLE5_periodic_advertising.ino b/libraries/BLE/examples/BLE5_periodic_advertising/BLE5_periodic_advertising.ino index 0b9d4f87630..effb7efd241 100644 --- a/libraries/BLE/examples/BLE5_periodic_advertising/BLE5_periodic_advertising.ino +++ b/libraries/BLE/examples/BLE5_periodic_advertising/BLE5_periodic_advertising.ino @@ -5,7 +5,9 @@ author: chegewara */ -#ifndef CONFIG_BT_BLE_50_FEATURES_SUPPORTED +#ifndef CONFIG_BLUEDROID_ENABLED +#error "NimBLE does not support periodic advertising yet. Try using Bluedroid." +#elif !defined(CONFIG_BT_BLE_50_FEATURES_SUPPORTED) #error "This SoC does not support BLE5. Try using ESP32-C3, or ESP32-S3" #else #include diff --git a/libraries/BLE/examples/BLE5_periodic_advertising/ci.json b/libraries/BLE/examples/BLE5_periodic_advertising/ci.json index 184cc25a2b0..8e938055cf2 100644 --- a/libraries/BLE/examples/BLE5_periodic_advertising/ci.json +++ b/libraries/BLE/examples/BLE5_periodic_advertising/ci.json @@ -1,6 +1,7 @@ { "fqbn_append": "PartitionScheme=huge_app", "requires": [ - "CONFIG_SOC_BLE_50_SUPPORTED=y" + "CONFIG_SOC_BLE_50_SUPPORTED=y", + "CONFIG_BLUEDROID_ENABLED=y" ] } diff --git a/libraries/BLE/examples/BLE5_periodic_sync/BLE5_periodic_sync.ino b/libraries/BLE/examples/BLE5_periodic_sync/BLE5_periodic_sync.ino index 9e976e6ca6a..a93ebcefec3 100644 --- a/libraries/BLE/examples/BLE5_periodic_sync/BLE5_periodic_sync.ino +++ b/libraries/BLE/examples/BLE5_periodic_sync/BLE5_periodic_sync.ino @@ -7,8 +7,10 @@ author: chegewara */ -#ifndef SOC_BLE_50_SUPPORTED -#warning "This SoC does not support BLE5. Try using ESP32-C3, or ESP32-S3" +#ifndef CONFIG_BLUEDROID_ENABLED +#error "NimBLE does not support periodic sync yet. Try using Bluedroid." +#elif !defined(SOC_BLE_50_SUPPORTED) +#error "This SoC does not support BLE5. Try using ESP32-C3, or ESP32-S3" #else #include #include diff --git a/libraries/BLE/examples/BLE5_periodic_sync/ci.json b/libraries/BLE/examples/BLE5_periodic_sync/ci.json index 184cc25a2b0..8e938055cf2 100644 --- a/libraries/BLE/examples/BLE5_periodic_sync/ci.json +++ b/libraries/BLE/examples/BLE5_periodic_sync/ci.json @@ -1,6 +1,7 @@ { "fqbn_append": "PartitionScheme=huge_app", "requires": [ - "CONFIG_SOC_BLE_50_SUPPORTED=y" + "CONFIG_SOC_BLE_50_SUPPORTED=y", + "CONFIG_BLUEDROID_ENABLED=y" ] } diff --git a/libraries/BLE/examples/Beacon_Scanner/Beacon_Scanner.ino b/libraries/BLE/examples/Beacon_Scanner/Beacon_Scanner.ino index fbbf89ad274..4b4a311a37d 100644 --- a/libraries/BLE/examples/Beacon_Scanner/Beacon_Scanner.ino +++ b/libraries/BLE/examples/Beacon_Scanner/Beacon_Scanner.ino @@ -36,25 +36,33 @@ class MyAdvertisedDeviceCallbacks : public BLEAdvertisedDeviceCallbacks { if (advertisedDevice.haveManufacturerData() == true) { String strManufacturerData = advertisedDevice.getManufacturerData(); - uint8_t cManufacturerData[100]; - memcpy(cManufacturerData, strManufacturerData.c_str(), strManufacturerData.length()); + // Buffer to store manufacturer data (BLE max is 255 bytes) + uint8_t cManufacturerData[255]; + size_t dataLength = strManufacturerData.length(); - if (strManufacturerData.length() == 25 && cManufacturerData[0] == 0x4C && cManufacturerData[1] == 0x00) { - Serial.println("Found an iBeacon!"); - BLEBeacon oBeacon = BLEBeacon(); - oBeacon.setData(strManufacturerData); - Serial.printf("iBeacon Frame\n"); - Serial.printf( - "ID: %04X Major: %d Minor: %d UUID: %s Power: %d\n", oBeacon.getManufacturerId(), ENDIAN_CHANGE_U16(oBeacon.getMajor()), - ENDIAN_CHANGE_U16(oBeacon.getMinor()), oBeacon.getProximityUUID().toString().c_str(), oBeacon.getSignalPower() - ); - } else { - Serial.println("Found another manufacturers beacon!"); - Serial.printf("strManufacturerData: %d ", strManufacturerData.length()); - for (int i = 0; i < strManufacturerData.length(); i++) { - Serial.printf("[%X]", cManufacturerData[i]); + // Bounds checking to prevent buffer overflow + if (dataLength <= sizeof(cManufacturerData)) { + memcpy(cManufacturerData, strManufacturerData.c_str(), dataLength); + + if (dataLength == 25 && cManufacturerData[0] == 0x4C && cManufacturerData[1] == 0x00) { + Serial.println("Found an iBeacon!"); + BLEBeacon oBeacon = BLEBeacon(); + oBeacon.setData(strManufacturerData); + Serial.printf("iBeacon Frame\n"); + Serial.printf( + "ID: %04X Major: %d Minor: %d UUID: %s Power: %d\n", oBeacon.getManufacturerId(), ENDIAN_CHANGE_U16(oBeacon.getMajor()), + ENDIAN_CHANGE_U16(oBeacon.getMinor()), oBeacon.getProximityUUID().toString().c_str(), oBeacon.getSignalPower() + ); + } else { + Serial.println("Found another manufacturers beacon!"); + Serial.printf("strManufacturerData: %d ", dataLength); + for (int i = 0; i < dataLength; i++) { + Serial.printf("[%X]", cManufacturerData[i]); + } + Serial.printf("\n"); } - Serial.printf("\n"); + } else { + Serial.printf("Manufacturer data too large (%d bytes), skipping\n", dataLength); } } diff --git a/libraries/BLE/examples/Client/Client.ino b/libraries/BLE/examples/Client/Client.ino index ce2eb2ff7d1..713bfda8e93 100644 --- a/libraries/BLE/examples/Client/Client.ino +++ b/libraries/BLE/examples/Client/Client.ino @@ -19,6 +19,7 @@ static boolean doScan = false; static BLERemoteCharacteristic *pRemoteCharacteristic; static BLEAdvertisedDevice *myDevice; +// Callback function to handle notifications static void notifyCallback(BLERemoteCharacteristic *pBLERemoteCharacteristic, uint8_t *pData, size_t length, bool isNotify) { Serial.print("Notify callback for characteristic "); Serial.print(pBLERemoteCharacteristic->getUUID().toString().c_str()); @@ -80,6 +81,7 @@ bool connectToServer() { } if (pRemoteCharacteristic->canNotify()) { + // Register/Subscribe for notifications pRemoteCharacteristic->registerForNotify(notifyCallback); } diff --git a/libraries/BLE/examples/Notify/Notify.ino b/libraries/BLE/examples/Notify/Notify.ino index 6b552b01d11..5f267685dfa 100644 --- a/libraries/BLE/examples/Notify/Notify.ino +++ b/libraries/BLE/examples/Notify/Notify.ino @@ -69,6 +69,7 @@ void setup() { ); // Creates BLE Descriptor 0x2902: Client Characteristic Configuration Descriptor (CCCD) + // Descriptor 2902 is not required when using NimBLE as it is automatically added based on the characteristic properties pCharacteristic->addDescriptor(new BLE2902()); // Adds also the Characteristic User Description - 0x2901 descriptor descriptor_2901 = new BLE2901(); diff --git a/libraries/BLE/examples/Server_multiconnect/Server_multiconnect.ino b/libraries/BLE/examples/Server_multiconnect/Server_multiconnect.ino index afd15e9ae4f..ff8712d8281 100644 --- a/libraries/BLE/examples/Server_multiconnect/Server_multiconnect.ino +++ b/libraries/BLE/examples/Server_multiconnect/Server_multiconnect.ino @@ -5,6 +5,9 @@ updated by chegewara Create a BLE server that, once we receive a connection, will send periodic notifications. + The server will continue advertising for more connections after the first one and will notify + the value of a counter to all connected clients. + The service advertises itself as: 4fafc201-1fb5-459e-8fcc-c5c9c331914b And has a characteristic of: beb5483e-36e1-4688-b7f5-ea07361b26a8 @@ -26,8 +29,8 @@ BLEServer *pServer = NULL; BLECharacteristic *pCharacteristic = NULL; +int connectedClients = 0; bool deviceConnected = false; -bool oldDeviceConnected = false; uint32_t value = 0; // See the following for generating UUIDs: @@ -38,12 +41,17 @@ uint32_t value = 0; class MyServerCallbacks : public BLEServerCallbacks { void onConnect(BLEServer *pServer) { - deviceConnected = true; + connectedClients++; + Serial.print("Client connected. Total clients: "); + Serial.println(connectedClients); + // Continue advertising for more connections BLEDevice::startAdvertising(); }; void onDisconnect(BLEServer *pServer) { - deviceConnected = false; + connectedClients--; + Serial.print("Client disconnected. Total clients: "); + Serial.println(connectedClients); } }; @@ -66,8 +74,7 @@ void setup() { BLECharacteristic::PROPERTY_READ | BLECharacteristic::PROPERTY_WRITE | BLECharacteristic::PROPERTY_NOTIFY | BLECharacteristic::PROPERTY_INDICATE ); - // https://www.bluetooth.com/specifications/gatt/viewer?attributeXmlFile=org.bluetooth.descriptor.gatt.client_characteristic_configuration.xml - // Create a BLE Descriptor + // Descriptor 2902 is not required when using NimBLE as it is automatically added based on the characteristic properties pCharacteristic->addDescriptor(new BLE2902()); // Start the service @@ -79,27 +86,38 @@ void setup() { pAdvertising->setScanResponse(false); pAdvertising->setMinPreferred(0x0); // set value to 0x00 to not advertise this parameter BLEDevice::startAdvertising(); - Serial.println("Waiting a client connection to notify..."); + Serial.println("Waiting for client connections to notify..."); } void loop() { - // notify changed value - if (deviceConnected) { + // Notify changed value to all connected clients + if (connectedClients > 0) { + Serial.print("Notifying value: "); + Serial.print(value); + Serial.print(" to "); + Serial.print(connectedClients); + Serial.println(" client(s)"); pCharacteristic->setValue((uint8_t *)&value, 4); pCharacteristic->notify(); value++; - delay(10); // bluetooth stack will go into congestion, if too many packets are sent, in 6 hours test i was able to go as low as 3ms + // Bluetooth stack will go into congestion, if too many packets are sent. + // In 6 hours of testing, I was able to go as low as 3ms. + // When using core debug level "debug" or "verbose", the delay can be increased in + // order to reduce the number of debug messages in the serial monitor. + delay(100); } - // disconnecting - if (!deviceConnected && oldDeviceConnected) { + + // Disconnecting - restart advertising when no clients are connected + if (connectedClients == 0 && deviceConnected) { delay(500); // give the bluetooth stack the chance to get things ready pServer->startAdvertising(); // restart advertising - Serial.println("start advertising"); - oldDeviceConnected = deviceConnected; + Serial.println("No clients connected, restarting advertising"); + deviceConnected = false; } - // connecting - if (deviceConnected && !oldDeviceConnected) { - // do stuff here on connecting - oldDeviceConnected = deviceConnected; + + // Connecting - update state when first client connects + if (connectedClients > 0 && !deviceConnected) { + // do stuff here on first connecting + deviceConnected = true; } } diff --git a/libraries/BLE/examples/UART/UART.ino b/libraries/BLE/examples/UART/UART.ino index 71cc850db66..d17ab494eca 100644 --- a/libraries/BLE/examples/UART/UART.ino +++ b/libraries/BLE/examples/UART/UART.ino @@ -40,10 +40,12 @@ uint8_t txValue = 0; class MyServerCallbacks : public BLEServerCallbacks { void onConnect(BLEServer *pServer) { deviceConnected = true; + Serial.println("Device connected"); }; void onDisconnect(BLEServer *pServer) { deviceConnected = false; + Serial.println("Device disconnected"); } }; @@ -80,6 +82,7 @@ void setup() { // Create a BLE Characteristic pTxCharacteristic = pService->createCharacteristic(CHARACTERISTIC_UUID_TX, BLECharacteristic::PROPERTY_NOTIFY); + // Descriptor 2902 is not required when using NimBLE as it is automatically added based on the characteristic properties pTxCharacteristic->addDescriptor(new BLE2902()); BLECharacteristic *pRxCharacteristic = pService->createCharacteristic(CHARACTERISTIC_UUID_RX, BLECharacteristic::PROPERTY_WRITE); @@ -97,22 +100,24 @@ void setup() { void loop() { if (deviceConnected) { + Serial.print("Notifying Value: "); + Serial.println(txValue); pTxCharacteristic->setValue(&txValue, 1); pTxCharacteristic->notify(); txValue++; - delay(10); // bluetooth stack will go into congestion, if too many packets are sent + delay(1000); // Notifying every 1 second } // disconnecting if (!deviceConnected && oldDeviceConnected) { delay(500); // give the bluetooth stack the chance to get things ready pServer->startAdvertising(); // restart advertising - Serial.println("start advertising"); - oldDeviceConnected = deviceConnected; + Serial.println("Started advertising again..."); + oldDeviceConnected = false; } // connecting if (deviceConnected && !oldDeviceConnected) { // do stuff here on connecting - oldDeviceConnected = deviceConnected; + oldDeviceConnected = true; } } diff --git a/libraries/BLE/library.properties b/libraries/BLE/library.properties index 7ef636223ec..fe39f5093c4 100644 --- a/libraries/BLE/library.properties +++ b/libraries/BLE/library.properties @@ -1,7 +1,7 @@ name=BLE version=3.2.0 author=Neil Kolban -maintainer=Dariusz Krempa +maintainer=lucasssvaz sentence=BLE functions for ESP32 paragraph=This library provides an implementation Bluetooth Low Energy support for the ESP32 using the Arduino platform. category=Communication diff --git a/libraries/BLE/src/BLE2901.cpp b/libraries/BLE/src/BLE2901.cpp index e929262b023..1fa4857ac33 100644 --- a/libraries/BLE/src/BLE2901.cpp +++ b/libraries/BLE/src/BLE2901.cpp @@ -1,29 +1,48 @@ /* - BLE2901.h + BLE2901.h - GATT Descriptor 0x2901 Characteristic User Description + GATT Descriptor 0x2901 Characteristic User Description - The value of this description is a user-readable string - describing the characteristic. + The value of this description is a user-readable string + describing the characteristic. - The Characteristic User Description descriptor - provides a textual user description for a characteristic - value. - If the Writable Auxiliary bit of the Characteristics - Properties is set then this descriptor is written. Only one - User Description descriptor exists in a characteristic - definition. + The Characteristic User Description descriptor + provides a textual user description for a characteristic + value. + If the Writable Auxiliary bit of the Characteristics + Properties is set then this descriptor is written. Only one + User Description descriptor exists in a characteristic + definition. */ #include "soc/soc_caps.h" #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) + +/*************************************************************************** + * Common includes and definitions * + ***************************************************************************/ #include "BLE2901.h" -BLE2901::BLE2901() : BLEDescriptor(BLEUUID((uint16_t)0x2901)) {} // BLE2901 +#define BLE2901_UUID 0x2901 + +/*************************************************************************** + * NimBLE includes and definitions * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) +#include +#define ESP_GATT_MAX_ATTR_LEN BLE_ATT_ATTR_MAX_LEN +#endif + +/*************************************************************************** + * Common functions * + ***************************************************************************/ + +BLE2901::BLE2901() : BLEDescriptor(BLEUUID((uint16_t)BLE2901_UUID)) {} /** * @brief Set the Characteristic User Description @@ -36,5 +55,5 @@ void BLE2901::setDescription(String userDesc) { setValue(userDesc); } -#endif +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ diff --git a/libraries/BLE/src/BLE2901.h b/libraries/BLE/src/BLE2901.h index f5ad7c94add..21e7cc9398c 100644 --- a/libraries/BLE/src/BLE2901.h +++ b/libraries/BLE/src/BLE2901.h @@ -1,37 +1,48 @@ /* - BLE2901.h + BLE2901.h - GATT Descriptor 0x2901 Characteristic User Description + GATT Descriptor 0x2901 Characteristic User Description - The value of this description is a user-readable string - describing the characteristic. - - The Characteristic User Description descriptor - provides a textual user description for a characteristic - value. - If the Writable Auxiliary bit of the Characteristics - Properties is set then this descriptor is written. Only one - User Description descriptor exists in a characteristic - definition. + The value of this description is a user-readable string + describing the characteristic. + The Characteristic User Description descriptor + provides a textual user description for a characteristic + value. + If the Writable Auxiliary bit of the Characteristics + Properties is set then this descriptor is written. Only one + User Description descriptor exists in a characteristic + definition. */ #ifndef COMPONENTS_CPP_UTILS_BLE2901_H_ #define COMPONENTS_CPP_UTILS_BLE2901_H_ + #include "soc/soc_caps.h" #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) + +/*************************************************************************** + * Common includes * + ***************************************************************************/ #include "BLEDescriptor.h" +/** + * @brief GATT Descriptor 0x2901 Characteristic User Description + */ class BLE2901 : public BLEDescriptor { public: + /*************************************************************************** + * Common public functions * + ***************************************************************************/ + BLE2901(); void setDescription(String desc); }; // BLE2901 -#endif /* CONFIG_BLUEDROID_ENABLED */ +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ #endif /* COMPONENTS_CPP_UTILS_BLE2901_H_ */ diff --git a/libraries/BLE/src/BLE2902.cpp b/libraries/BLE/src/BLE2902.cpp index 90cdf088ff2..3bac9281328 100644 --- a/libraries/BLE/src/BLE2902.cpp +++ b/libraries/BLE/src/BLE2902.cpp @@ -3,46 +3,86 @@ * * Created on: Jun 25, 2017 * Author: kolban + * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on kolban's and h2zero's work) + * Description: Added support for NimBLE */ /* * See also: * https://www.bluetooth.com/specifications/gatt/viewer?attributeXmlFile=org.bluetooth.descriptor.gatt.client_characteristic_configuration.xml */ + #include "soc/soc_caps.h" #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) + +/*************************************************************************** + * Common includes and definitions * + ***************************************************************************/ #include "BLE2902.h" -BLE2902::BLE2902() : BLEDescriptor(BLEUUID((uint16_t)0x2902)) { +#define BLE2902_UUID 0x2902 + +/*************************************************************************** + * Common functions * + ***************************************************************************/ + +BLE2902::BLE2902() : BLEDescriptor(BLEUUID((uint16_t)BLE2902_UUID)) { +#if defined(CONFIG_BLUEDROID_ENABLED) uint8_t data[2] = {0, 0}; setValue(data, 2); -} // BLE2902 +#endif +} /** * @brief Get the notifications value. * @return The notifications value. True if notifications are enabled and false if not. */ bool BLE2902::getNotifications() { +#if defined(CONFIG_BLUEDROID_ENABLED) return (getValue()[0] & (1 << 0)) != 0; -} // getNotifications +#endif + +#if defined(CONFIG_NIMBLE_ENABLED) + if (m_pCharacteristic != nullptr) { + return (m_pCharacteristic->getProperties() & BLECharacteristic::PROPERTY_NOTIFY) != 0; + } else { + log_w("BLE2902::getNotifications() called on an uninitialized descriptor"); + return false; + } +#endif +} /** * @brief Get the indications value. * @return The indications value. True if indications are enabled and false if not. */ bool BLE2902::getIndications() { +#if defined(CONFIG_BLUEDROID_ENABLED) return (getValue()[0] & (1 << 1)) != 0; -} // getIndications +#endif + +#if defined(CONFIG_NIMBLE_ENABLED) + if (m_pCharacteristic != nullptr) { + return (m_pCharacteristic->getProperties() & BLECharacteristic::PROPERTY_INDICATE) != 0; + } else { + log_w("BLE2902::getIndications() called on an uninitialized descriptor"); + return false; + } +#endif +} /** * @brief Set the indications flag. * @param [in] flag The indications flag. */ void BLE2902::setIndications(bool flag) { +#if defined(CONFIG_BLUEDROID_ENABLED) uint8_t *pValue = getValue(); if (flag) { pValue[0] |= 1 << 1; @@ -50,13 +90,23 @@ void BLE2902::setIndications(bool flag) { pValue[0] &= ~(1 << 1); } setValue(pValue, 2); -} // setIndications +#endif + +#if defined(CONFIG_NIMBLE_ENABLED) + if (m_pCharacteristic != nullptr) { + m_pCharacteristic->setIndicateProperty(flag); + } else { + log_w("BLE2902::setIndications() called on an uninitialized descriptor"); + } +#endif +} /** * @brief Set the notifications flag. * @param [in] flag The notifications flag. */ void BLE2902::setNotifications(bool flag) { +#if defined(CONFIG_BLUEDROID_ENABLED) uint8_t *pValue = getValue(); if (flag) { pValue[0] |= 1 << 0; @@ -64,7 +114,16 @@ void BLE2902::setNotifications(bool flag) { pValue[0] &= ~(1 << 0); } setValue(pValue, 2); -} // setNotifications +#endif +#if defined(CONFIG_NIMBLE_ENABLED) + if (m_pCharacteristic != nullptr) { + m_pCharacteristic->setNotifyProperty(flag); + } else { + log_w("BLE2902::setNotifications() called on an uninitialized descriptor"); + } #endif +} + +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ diff --git a/libraries/BLE/src/BLE2902.h b/libraries/BLE/src/BLE2902.h index 74a477f3151..5cbf911ea4a 100644 --- a/libraries/BLE/src/BLE2902.h +++ b/libraries/BLE/src/BLE2902.h @@ -3,15 +3,24 @@ * * Created on: Jun 25, 2017 * Author: kolban + * + * Modified on: Feb 28, 2025 + * Author: lucasssvaz (based on kolban's and h2zero's work) + * Description: Added support for NimBLE */ #ifndef COMPONENTS_CPP_UTILS_BLE2902_H_ #define COMPONENTS_CPP_UTILS_BLE2902_H_ + #include "soc/soc_caps.h" #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) + +/*************************************************************************** + * Common includes * + ***************************************************************************/ #include "BLEDescriptor.h" @@ -23,16 +32,35 @@ * See also: * https://www.bluetooth.com/specifications/gatt/viewer?attributeXmlFile=org.bluetooth.descriptor.gatt.client_characteristic_configuration.xml */ + +// Class declaration for Bluedroid +#if defined(CONFIG_BLUEDROID_ENABLED) class BLE2902 : public BLEDescriptor { -public: - BLE2902(); - bool getNotifications(); - bool getIndications(); - void setNotifications(bool flag); - void setIndications(bool flag); +#endif + +// Class declaration for NimBLE (deprecated) +#if defined(CONFIG_NIMBLE_ENABLED) + class [[deprecated("NimBLE does not support manually adding 2902 descriptors as they \ +are automatically added when the characteristic has notifications or indications enabled. \ +Get/Set the notifications/indications properties of the characteristic instead. \ +This class will be removed in a future version.")]] BLE2902 : public BLEDescriptor { +#endif + + public: + /*************************************************************************** + * Common public functions * + ***************************************************************************/ + + BLE2902(); + bool getNotifications(); + bool getIndications(); + void setNotifications(bool flag); + void setIndications(bool flag); -}; // BLE2902 + private: + friend class BLECharacteristic; + }; // BLE2902 -#endif /* CONFIG_BLUEDROID_ENABLED */ +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ #endif /* COMPONENTS_CPP_UTILS_BLE2902_H_ */ diff --git a/libraries/BLE/src/BLE2904.cpp b/libraries/BLE/src/BLE2904.cpp index aeed11ebad1..19658051120 100644 --- a/libraries/BLE/src/BLE2904.cpp +++ b/libraries/BLE/src/BLE2904.cpp @@ -3,6 +3,10 @@ * * Created on: Dec 23, 2017 * Author: kolban + * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on kolban's and h2zero's work) + * Description: Added support for NimBLE */ /* @@ -13,18 +17,30 @@ #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) + +/*************************************************************************** + * Common includes and definitions * + ***************************************************************************/ #include "BLE2904.h" -BLE2904::BLE2904() : BLEDescriptor(BLEUUID((uint16_t)0x2904)) { +#define BLE2904_UUID 0x2904 +#define BLE2904_DEFAULT_NAMESPACE 1 // 1 = Bluetooth SIG Assigned Numbers +#define BLE2904_DEFAULT_UNIT 0x2700 // 0x2700 = Unitless + +/*************************************************************************** + * Common functions * + ***************************************************************************/ + +BLE2904::BLE2904() : BLEDescriptor(BLEUUID((uint16_t)BLE2904_UUID)) { m_data.m_format = 0; m_data.m_exponent = 0; - m_data.m_namespace = 1; // 1 = Bluetooth SIG Assigned Numbers - m_data.m_unit = 0; + m_data.m_namespace = BLE2904_DEFAULT_NAMESPACE; + m_data.m_unit = BLE2904_DEFAULT_UNIT; m_data.m_description = 0; setValue((uint8_t *)&m_data, sizeof(m_data)); -} // BLE2902 +} /** * @brief Set the description. @@ -40,7 +56,7 @@ void BLE2904::setDescription(uint16_t description) { void BLE2904::setExponent(int8_t exponent) { m_data.m_exponent = exponent; setValue((uint8_t *)&m_data, sizeof(m_data)); -} // setExponent +} /** * @brief Set the format. @@ -48,7 +64,7 @@ void BLE2904::setExponent(int8_t exponent) { void BLE2904::setFormat(uint8_t format) { m_data.m_format = format; setValue((uint8_t *)&m_data, sizeof(m_data)); -} // setFormat +} /** * @brief Set the namespace. @@ -56,7 +72,7 @@ void BLE2904::setFormat(uint8_t format) { void BLE2904::setNamespace(uint8_t namespace_value) { m_data.m_namespace = namespace_value; setValue((uint8_t *)&m_data, sizeof(m_data)); -} // setNamespace +} /** * @brief Set the units for this value. It should be one of the encoded values defined here: @@ -66,7 +82,7 @@ void BLE2904::setNamespace(uint8_t namespace_value) { void BLE2904::setUnit(uint16_t unit) { m_data.m_unit = unit; setValue((uint8_t *)&m_data, sizeof(m_data)); -} // setUnit +} -#endif +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ diff --git a/libraries/BLE/src/BLE2904.h b/libraries/BLE/src/BLE2904.h index 3ba66da0dc8..87e8c03c048 100644 --- a/libraries/BLE/src/BLE2904.h +++ b/libraries/BLE/src/BLE2904.h @@ -3,27 +3,43 @@ * * Created on: Dec 23, 2017 * Author: kolban + * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on kolban's and h2zero's work) + * Description: Added support for NimBLE */ #ifndef COMPONENTS_CPP_UTILS_BLE2904_H_ #define COMPONENTS_CPP_UTILS_BLE2904_H_ + #include "soc/soc_caps.h" #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) + +/*************************************************************************** + * Common includes * + ***************************************************************************/ #include "BLEDescriptor.h" +/*************************************************************************** + * Common types * + ***************************************************************************/ + struct BLE2904_Data { uint8_t m_format; int8_t m_exponent; uint16_t m_unit; // See https://www.bluetooth.com/specifications/assigned-numbers/units uint8_t m_namespace; uint16_t m_description; - } __attribute__((packed)); +/*************************************************************************** + * Common functions * + ***************************************************************************/ + /** * @brief Descriptor for Characteristic Presentation Format. * @@ -34,7 +50,10 @@ struct BLE2904_Data { */ class BLE2904 : public BLEDescriptor { public: - BLE2904(); + /*************************************************************************** + * Common public constants * + ***************************************************************************/ + static const uint8_t FORMAT_BOOLEAN = 1; static const uint8_t FORMAT_UINT2 = 2; static const uint8_t FORMAT_UINT4 = 3; @@ -62,7 +81,13 @@ class BLE2904 : public BLEDescriptor { static const uint8_t FORMAT_UTF8 = 25; static const uint8_t FORMAT_UTF16 = 26; static const uint8_t FORMAT_OPAQUE = 27; + static const uint8_t FORMAT_MEDASN1 = 28; + + /*************************************************************************** + * Common public functions * + ***************************************************************************/ + BLE2904(); void setDescription(uint16_t); void setExponent(int8_t exponent); void setFormat(uint8_t format); @@ -70,9 +95,15 @@ class BLE2904 : public BLEDescriptor { void setUnit(uint16_t unit); private: + friend class BLECharacteristic; + + /*************************************************************************** + * Common private properties * + ***************************************************************************/ + BLE2904_Data m_data; }; // BLE2904 -#endif /* CONFIG_BLUEDROID_ENABLED */ +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ #endif /* COMPONENTS_CPP_UTILS_BLE2904_H_ */ diff --git a/libraries/BLE/src/BLEAddress.cpp b/libraries/BLE/src/BLEAddress.cpp index b91ef3cc4de..11c64850367 100644 --- a/libraries/BLE/src/BLEAddress.cpp +++ b/libraries/BLE/src/BLEAddress.cpp @@ -3,12 +3,21 @@ * * Created on: Jul 2, 2017 * Author: kolban + * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on kolban's and h2zero's work) + * Description: Added support for NimBLE */ + #include "soc/soc_caps.h" #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) + +/*************************************************************************** + * Common includes * + ***************************************************************************/ #include "BLEAddress.h" #include @@ -17,43 +26,33 @@ #include #include #include + #ifdef ARDUINO_ARCH_ESP32 #include "esp32-hal-log.h" #endif -/** - * @brief Create an address from the native ESP32 representation. - * @param [in] address The native representation. - */ -BLEAddress::BLEAddress(esp_bd_addr_t address) { - memcpy(m_address, address, ESP_BD_ADDR_LEN); -} // BLEAddress +/*************************************************************************** + * NimBLE includes * + ***************************************************************************/ -/** - * @brief Create an address from a hex string - * - * A hex string is of the format: - * ``` - * 00:00:00:00:00:00 - * ``` - * which is 17 characters in length. - * - * @param [in] stringAddress The hex representation of the address. - */ -BLEAddress::BLEAddress(String stringAddress) { - if (stringAddress.length() != 17) { - return; - } +#if defined(CONFIG_NIMBLE_ENABLED) +/************************************************* + * NOTE: NimBLE address bytes are in INVERSE ORDER! + * We will accommodate that fact in these methods. +*************************************************/ +#include +#endif - int data[6]; - sscanf(stringAddress.c_str(), "%x:%x:%x:%x:%x:%x", &data[0], &data[1], &data[2], &data[3], &data[4], &data[5]); - m_address[0] = (uint8_t)data[0]; - m_address[1] = (uint8_t)data[1]; - m_address[2] = (uint8_t)data[2]; - m_address[3] = (uint8_t)data[3]; - m_address[4] = (uint8_t)data[4]; - m_address[5] = (uint8_t)data[5]; -} // BLEAddress +/*************************************************************************** + * Common functions * + ***************************************************************************/ + +BLEAddress::BLEAddress() { + memset(m_address, 0, ESP_BD_ADDR_LEN); +#if defined(CONFIG_NIMBLE_ENABLED) + m_addrType = 0; +#endif +} /** * @brief Determine if this address equals another. @@ -61,10 +60,20 @@ BLEAddress::BLEAddress(String stringAddress) { * @return True if the addresses are equal. */ bool BLEAddress::equals(BLEAddress otherAddress) { +#if defined(CONFIG_NIMBLE_ENABLED) + if (m_addrType != otherAddress.m_addrType) { + return false; + } +#endif return memcmp(otherAddress.getNative(), m_address, ESP_BD_ADDR_LEN) == 0; -} // equals +} bool BLEAddress::operator==(const BLEAddress &otherAddress) const { +#if defined(CONFIG_NIMBLE_ENABLED) + if (m_addrType != otherAddress.m_addrType) { + return false; + } +#endif return memcmp(otherAddress.m_address, m_address, ESP_BD_ADDR_LEN) == 0; } @@ -92,9 +101,9 @@ bool BLEAddress::operator>(const BLEAddress &otherAddress) const { * @brief Return the native representation of the address. * @return The native representation of the address. */ -esp_bd_addr_t *BLEAddress::getNative() { - return &m_address; -} // getNative +uint8_t *BLEAddress::getNative() { + return m_address; +} /** * @brief Convert a BLE address to a string. @@ -110,11 +119,112 @@ esp_bd_addr_t *BLEAddress::getNative() { String BLEAddress::toString() { auto size = 18; char *res = (char *)malloc(size); + +#if defined(CONFIG_BLUEDROID_ENABLED) snprintf(res, size, "%02x:%02x:%02x:%02x:%02x:%02x", m_address[0], m_address[1], m_address[2], m_address[3], m_address[4], m_address[5]); +#endif + +#if defined(CONFIG_NIMBLE_ENABLED) + snprintf(res, size, "%02x:%02x:%02x:%02x:%02x:%02x", m_address[5], m_address[4], m_address[3], m_address[2], m_address[1], m_address[0]); +#endif + String ret(res); free(res); return ret; -} // toString +} + +/*************************************************************************** + * Bluedroid functions * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) + +/** + * @brief Create an address from the native ESP32 representation. + * @param [in] address The native representation. + */ +BLEAddress::BLEAddress(esp_bd_addr_t address) { + memcpy(m_address, address, ESP_BD_ADDR_LEN); +} + +/** + * @brief Create an address from a hex string + * + * A hex string is of the format: + * ``` + * 00:00:00:00:00:00 + * ``` + * which is 17 characters in length. + * + * @param [in] stringAddress The hex representation of the address. + */ +BLEAddress::BLEAddress(String stringAddress) { + if (stringAddress.length() != 17) { + return; + } + + int data[6]; + sscanf(stringAddress.c_str(), "%x:%x:%x:%x:%x:%x", &data[0], &data[1], &data[2], &data[3], &data[4], &data[5]); + + for (size_t index = 0; index < sizeof(m_address); index++) { + m_address[index] = (uint8_t)data[index]; + } +} + +#endif + +/*************************************************************************** + * NimBLE functions * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) +/************************************************* + * NOTE: NimBLE address bytes are in INVERSE ORDER! + * We will accommodate that fact in these methods. +*************************************************/ + +BLEAddress::BLEAddress(uint8_t address[ESP_BD_ADDR_LEN], uint8_t type) { + std::reverse_copy(address, address + sizeof(m_address), m_address); + m_addrType = type; +} + +BLEAddress::BLEAddress(ble_addr_t address) { + memcpy(m_address, address.val, ESP_BD_ADDR_LEN); + m_addrType = address.type; +} + +uint8_t BLEAddress::getType() { + return m_addrType; +} + +/** + * @brief Create an address from a hex string + * + * A hex string is of the format: + * ``` + * 00:00:00:00:00:00 + * ``` + * which is 17 characters in length. + * + * @param [in] stringAddress The hex representation of the address. + * @param [in] type The address type. + */ +BLEAddress::BLEAddress(String stringAddress, uint8_t type) { + if (stringAddress.length() != 17) { + return; + } + + int data[6]; + m_addrType = type; + // NimBLE addresses are in INVERSE ORDER! + sscanf(stringAddress.c_str(), "%x:%x:%x:%x:%x:%x", &data[5], &data[4], &data[3], &data[2], &data[1], &data[0]); + + for (size_t index = 0; index < sizeof(m_address); index++) { + m_address[index] = (uint8_t)data[index]; + } +} #endif + +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ diff --git a/libraries/BLE/src/BLEAddress.h b/libraries/BLE/src/BLEAddress.h index f1c8aa9b632..71b3bdfce5f 100644 --- a/libraries/BLE/src/BLEAddress.h +++ b/libraries/BLE/src/BLEAddress.h @@ -3,19 +3,53 @@ * * Created on: Jul 2, 2017 * Author: kolban + * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on kolban's and h2zero's work) + * Description: Added support for NimBLE */ #ifndef COMPONENTS_CPP_UTILS_BLEADDRESS_H_ #define COMPONENTS_CPP_UTILS_BLEADDRESS_H_ #include "soc/soc_caps.h" -#include "WString.h" #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) -#include // ESP32 BLE +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) + +/*************************************************************************** + * Common includes * + ***************************************************************************/ + +#include "WString.h" +#include #include +/*************************************************************************** + * Bluedroid includes * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) +#include +#endif + +/*************************************************************************** + * NimBLE includes and definitions * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) +#include +#define ESP_BD_ADDR_LEN BLE_DEV_ADDR_LEN +#endif + +/*************************************************************************** + * NimBLE types * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) +typedef uint8_t esp_bd_addr_t[ESP_BD_ADDR_LEN]; +#endif + /** * @brief A %BLE device address. * @@ -23,8 +57,11 @@ */ class BLEAddress { public: - BLEAddress(esp_bd_addr_t address); - BLEAddress(String stringAddress); + /*************************************************************************** + * Common public declarations * + ***************************************************************************/ + + BLEAddress(); bool equals(BLEAddress otherAddress); bool operator==(const BLEAddress &otherAddress) const; bool operator!=(const BLEAddress &otherAddress) const; @@ -32,13 +69,45 @@ class BLEAddress { bool operator<=(const BLEAddress &otherAddress) const; bool operator>(const BLEAddress &otherAddress) const; bool operator>=(const BLEAddress &otherAddress) const; - esp_bd_addr_t *getNative(); + uint8_t *getNative(); String toString(); + /*************************************************************************** + * Bluedroid public declarations * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) + BLEAddress(esp_bd_addr_t address); + BLEAddress(String stringAddress); +#endif + + /*************************************************************************** + * NimBLE public declarations * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) + BLEAddress(ble_addr_t address); + BLEAddress(String stringAddress, uint8_t type = BLE_ADDR_PUBLIC); + BLEAddress(uint8_t address[ESP_BD_ADDR_LEN], uint8_t type = BLE_ADDR_PUBLIC); + uint8_t getType(); +#endif + private: - esp_bd_addr_t m_address; + /*************************************************************************** + * Common private properties * + ***************************************************************************/ + + uint8_t m_address[ESP_BD_ADDR_LEN]; + + /*************************************************************************** + * NimBLE private properties * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) + uint8_t m_addrType; +#endif }; -#endif /* CONFIG_BLUEDROID_ENABLED */ +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ #endif /* COMPONENTS_CPP_UTILS_BLEADDRESS_H_ */ diff --git a/libraries/BLE/src/BLEAdvertisedDevice.cpp b/libraries/BLE/src/BLEAdvertisedDevice.cpp index 8752d24a199..3ac1dfab5c8 100644 --- a/libraries/BLE/src/BLEAdvertisedDevice.cpp +++ b/libraries/BLE/src/BLEAdvertisedDevice.cpp @@ -4,22 +4,35 @@ * During the scanning procedure, we will be finding advertised BLE devices. This class * models a found device. * - * * See also: * https://www.bluetooth.com/specifications/assigned-numbers/generic-access-profile * * Created on: Jul 3, 2017 * Author: kolban + * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on kolban's and h2zero's work) + * Description: Added support for NimBLE */ + #include "soc/soc_caps.h" #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) + +/*************************************************************************** + * Common includes * + ***************************************************************************/ + #include #include "BLEAdvertisedDevice.h" #include "BLEUtils.h" #include "esp32-hal-log.h" +/*************************************************************************** + * Common functions * + ***************************************************************************/ + BLEAdvertisedDevice::BLEAdvertisedDevice() { m_adFlag = 0; m_appearance = 0; @@ -32,15 +45,47 @@ BLEAdvertisedDevice::BLEAdvertisedDevice() { m_serviceDataUUIDs = {}; m_txPower = 0; m_pScan = nullptr; + m_advType = 0; + +#if defined(CONFIG_NIMBLE_ENABLED) + m_callbackSent = false; +#endif m_haveAppearance = false; m_haveManufacturerData = false; m_haveName = false; m_haveRSSI = false; m_haveTXPower = false; - + m_isLegacyAdv = true; } // BLEAdvertisedDevice +bool BLEAdvertisedDevice::isLegacyAdvertisement() { + return m_isLegacyAdv; +} + +bool BLEAdvertisedDevice::isScannable() { +#if defined(CONFIG_BLUEDROID_ENABLED) + return isLegacyAdvertisement() && (m_advType == ESP_BLE_EVT_CONN_ADV || m_advType == ESP_BLE_EVT_DISC_ADV); +#endif + +#if defined(CONFIG_NIMBLE_ENABLED) + return isLegacyAdvertisement() && (m_advType == BLE_HCI_ADV_TYPE_ADV_IND || m_advType == BLE_HCI_ADV_TYPE_ADV_SCAN_IND); +#endif +} + +bool BLEAdvertisedDevice::isConnectable() { +#if defined(CONFIG_BLUEDROID_ENABLED) + return m_advType == ESP_BLE_EVT_CONN_ADV || m_advType == ESP_BLE_EVT_CONN_DIR_ADV; +#endif + +#if defined(CONFIG_NIMBLE_ENABLED) + if (m_isLegacyAdv) { + return m_advType == BLE_HCI_ADV_RPT_EVTYPE_ADV_IND || m_advType == BLE_HCI_ADV_RPT_EVTYPE_DIR_IND; + } + return (m_advType & BLE_HCI_ADV_CONN_MASK) || (m_advType & BLE_HCI_ADV_DIRECT_MASK); +#endif +} + /** * @brief Get the address. * @@ -277,75 +322,75 @@ void BLEAdvertisedDevice::parseAdvertisement(uint8_t *payload, size_t total_len) length--; char *pHex = BLEUtils::buildHexData(nullptr, payload, length); - log_d("Type: 0x%.2x (%s), length: %d, data: %s", ad_type, BLEUtils::advTypeToString(ad_type), length, pHex); + log_d("Type: 0x%.2x (%s), length: %d, data: %s", ad_type, BLEUtils::advDataTypeToString(ad_type), length, pHex); free(pHex); switch (ad_type) { - case ESP_BLE_AD_TYPE_NAME_CMPL: - { // Adv Data Type: 0x09 + case ESP_BLE_AD_TYPE_NAME_CMPL: // 0x09 + { // Adv Data Type: ESP_BLE_AD_TYPE_NAME_CMPL setName(String(reinterpret_cast(payload), length)); break; - } // ESP_BLE_AD_TYPE_NAME_CMPL + } // 0x09 - case ESP_BLE_AD_TYPE_TX_PWR: - { // Adv Data Type: 0x0A + case ESP_BLE_AD_TYPE_TX_PWR: // 0x0A + { // Adv Data Type: ESP_BLE_AD_TYPE_TX_PWR setTXPower(*payload); break; - } // ESP_BLE_AD_TYPE_TX_PWR + } // 0x0A - case ESP_BLE_AD_TYPE_APPEARANCE: - { // Adv Data Type: 0x19 + case ESP_BLE_AD_TYPE_APPEARANCE: // 0x19 + { // Adv Data Type: ESP_BLE_AD_TYPE_APPEARANCE setAppearance(*reinterpret_cast(payload)); break; - } // ESP_BLE_AD_TYPE_APPEARANCE + } // 0x19 - case ESP_BLE_AD_TYPE_FLAG: - { // Adv Data Type: 0x01 + case ESP_BLE_AD_TYPE_FLAG: // 0x01 + { // Adv Data Type: ESP_BLE_AD_TYPE_FLAG setAdFlag(*payload); break; - } // ESP_BLE_AD_TYPE_FLAG + } // 0x01 - case ESP_BLE_AD_TYPE_16SRV_CMPL: - case ESP_BLE_AD_TYPE_16SRV_PART: - { // Adv Data Type: 0x02 + case ESP_BLE_AD_TYPE_16SRV_PART: // 0x02 + case ESP_BLE_AD_TYPE_16SRV_CMPL: // 0x03 + { // Adv Data Type: ESP_BLE_AD_TYPE_16SRV_PART/CMPL for (int var = 0; var < length / 2; ++var) { setServiceUUID(BLEUUID(*reinterpret_cast(payload + var * 2))); } break; - } // ESP_BLE_AD_TYPE_16SRV_PART + } // 0x02, 0x03 - case ESP_BLE_AD_TYPE_32SRV_CMPL: - case ESP_BLE_AD_TYPE_32SRV_PART: - { // Adv Data Type: 0x04 + case ESP_BLE_AD_TYPE_32SRV_PART: // 0x04 + case ESP_BLE_AD_TYPE_32SRV_CMPL: // 0x05 + { // Adv Data Type: ESP_BLE_AD_TYPE_32SRV_PART/CMPL for (int var = 0; var < length / 4; ++var) { setServiceUUID(BLEUUID(*reinterpret_cast(payload + var * 4))); } break; - } // ESP_BLE_AD_TYPE_32SRV_PART + } // 0x04, 0x05 - case ESP_BLE_AD_TYPE_128SRV_CMPL: - { // Adv Data Type: 0x07 + case ESP_BLE_AD_TYPE_128SRV_CMPL: // 0x07 + { // Adv Data Type: ESP_BLE_AD_TYPE_128SRV_CMPL setServiceUUID(BLEUUID(payload, 16, false)); break; - } // ESP_BLE_AD_TYPE_128SRV_CMPL + } // 0x07 - case ESP_BLE_AD_TYPE_128SRV_PART: - { // Adv Data Type: 0x06 + case ESP_BLE_AD_TYPE_128SRV_PART: // 0x06 + { // Adv Data Type: ESP_BLE_AD_TYPE_128SRV_PART setServiceUUID(BLEUUID(payload, 16, false)); break; - } // ESP_BLE_AD_TYPE_128SRV_PART + } // 0x06 // See CSS Part A 1.4 Manufacturer Specific Data - case ESP_BLE_AD_MANUFACTURER_SPECIFIC_TYPE: + case ESP_BLE_AD_MANUFACTURER_SPECIFIC_TYPE: // 0xFF { setManufacturerData(String(reinterpret_cast(payload), length)); break; - } // ESP_BLE_AD_MANUFACTURER_SPECIFIC_TYPE + } // 0xFF - case ESP_BLE_AD_TYPE_SERVICE_DATA: - { // Adv Data Type: 0x16 (Service Data) - 2 byte UUID + case ESP_BLE_AD_TYPE_SERVICE_DATA: // 0x16 + { // Adv Data Type: ESP_BLE_AD_TYPE_SERVICE_DATA - 2 byte UUID if (length < 2) { - log_e("Length too small for ESP_BLE_AD_TYPE_SERVICE_DATA"); + log_e("Length too small for SERVICE_DATA"); break; } uint16_t uuid = *(uint16_t *)payload; @@ -354,12 +399,12 @@ void BLEAdvertisedDevice::parseAdvertisement(uint8_t *payload, size_t total_len) setServiceData(String(reinterpret_cast(payload + 2), length - 2)); } break; - } //ESP_BLE_AD_TYPE_SERVICE_DATA + } // 0x16 - case ESP_BLE_AD_TYPE_32SERVICE_DATA: - { // Adv Data Type: 0x20 (Service Data) - 4 byte UUID + case ESP_BLE_AD_TYPE_32SERVICE_DATA: // 0x20 + { // Adv Data Type: ESP_BLE_AD_TYPE_32SERVICE_DATA - 4 byte UUID if (length < 4) { - log_e("Length too small for ESP_BLE_AD_TYPE_32SERVICE_DATA"); + log_e("Length too small for 32SERVICE_DATA"); break; } uint32_t uuid = *(uint32_t *)payload; @@ -368,12 +413,12 @@ void BLEAdvertisedDevice::parseAdvertisement(uint8_t *payload, size_t total_len) setServiceData(String(reinterpret_cast(payload + 4), length - 4)); } break; - } //ESP_BLE_AD_TYPE_32SERVICE_DATA + } // 0x20 - case ESP_BLE_AD_TYPE_128SERVICE_DATA: - { // Adv Data Type: 0x21 (Service Data) - 16 byte UUID + case ESP_BLE_AD_TYPE_128SERVICE_DATA: // 0x21 + { // Adv Data Type: ESP_BLE_AD_TYPE_128SERVICE_DATA - 16 byte UUID if (length < 16) { - log_e("Length too small for ESP_BLE_AD_TYPE_128SERVICE_DATA"); + log_e("Length too small for 128SERVICE_DATA"); break; } @@ -382,13 +427,13 @@ void BLEAdvertisedDevice::parseAdvertisement(uint8_t *payload, size_t total_len) setServiceData(String(reinterpret_cast(payload + 16), length - 16)); } break; - } //ESP_BLE_AD_TYPE_32SERVICE_DATA + } // 0x21 default: { log_d("Unhandled type: adType: %d - 0x%.2x", ad_type, ad_type); break; - } + } // default } // switch payload += length; } // Length <> 0 @@ -405,9 +450,19 @@ void BLEAdvertisedDevice::parseAdvertisement(uint8_t *payload, size_t total_len) * @param [in] payload The payload of the advertised device. * @param [in] total_len The length of payload */ -void BLEAdvertisedDevice::setPayload(uint8_t *payload, size_t total_len) { - m_payload = payload; - m_payloadLength = total_len; +void BLEAdvertisedDevice::setPayload(uint8_t *payload, size_t total_len, bool append) { + if (m_payload == nullptr || m_payloadLength == 0) { + return; + } + + if (append) { + m_payload = (uint8_t *)realloc(m_payload, m_payloadLength + total_len); + memcpy(m_payload + m_payloadLength, payload, total_len); + m_payloadLength += total_len; + } else { + m_payload = payload; + m_payloadLength = total_len; + } } // setPayload /** @@ -567,7 +622,7 @@ uint8_t *BLEAdvertisedDevice::getPayload() { return m_payload; } -esp_ble_addr_type_t BLEAdvertisedDevice::getAddressType() { +uint8_t BLEAdvertisedDevice::getAddressType() { return m_addressType; } @@ -587,7 +642,7 @@ ble_frame_type_t BLEAdvertisedDevice::getFrameType() { return BLE_UNKNOWN_FRAME; } -void BLEAdvertisedDevice::setAddressType(esp_ble_addr_type_t type) { +void BLEAdvertisedDevice::setAddressType(uint8_t type) { m_addressType = type; } @@ -595,5 +650,13 @@ size_t BLEAdvertisedDevice::getPayloadLength() { return m_payloadLength; } -#endif /* CONFIG_BLUEDROID_ENABLED */ +void BLEAdvertisedDevice::setAdvType(uint8_t type) { + m_advType = type; +} + +uint8_t BLEAdvertisedDevice::getAdvType() { + return m_advType; +} + +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ diff --git a/libraries/BLE/src/BLEAdvertisedDevice.h b/libraries/BLE/src/BLEAdvertisedDevice.h index 700e5704034..2a2fc0c81aa 100644 --- a/libraries/BLE/src/BLEAdvertisedDevice.h +++ b/libraries/BLE/src/BLEAdvertisedDevice.h @@ -3,6 +3,10 @@ * * Created on: Jul 3, 2017 * Author: kolban + * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on kolban's and h2zero's work) + * Description: Added support for NimBLE */ #ifndef COMPONENTS_CPP_UTILS_BLEADVERTISEDDEVICE_H_ @@ -11,15 +15,38 @@ #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) -#include +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) -#include +/*************************************************************************** + * Common includes * + ***************************************************************************/ +#include #include "BLEAddress.h" #include "BLEScan.h" #include "BLEUUID.h" +/*************************************************************************** + * Bluedroid includes * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) +#include +#endif + +/*************************************************************************** + * NimBLE includes * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) +#include +#include +#endif + +/*************************************************************************** + * Common types * + ***************************************************************************/ + typedef enum { BLE_UNKNOWN_FRAME, BLE_EDDYSTONE_UUID_FRAME, @@ -28,7 +55,12 @@ typedef enum { BLE_FRAME_MAX } ble_frame_type_t; +/*************************************************************************** + * Forward declarations * + ***************************************************************************/ + class BLEScan; + /** * @brief A representation of a %BLE advertised device found by a scan. * @@ -37,8 +69,11 @@ class BLEScan; */ class BLEAdvertisedDevice { public: - BLEAdvertisedDevice(); + /*************************************************************************** + * Common public declarations * + ***************************************************************************/ + BLEAdvertisedDevice(); BLEAddress getAddress(); uint16_t getAppearance(); String getManufacturerData(); @@ -57,10 +92,15 @@ class BLEAdvertisedDevice { int8_t getTXPower(); uint8_t *getPayload(); size_t getPayloadLength(); - esp_ble_addr_type_t getAddressType(); + uint8_t getAddressType(); ble_frame_type_t getFrameType(); - void setAddressType(esp_ble_addr_type_t type); + void setAddressType(uint8_t type); + void setAdvType(uint8_t type); + uint8_t getAdvType(); + bool isLegacyAdvertisement(); + bool isScannable(); + bool isConnectable(); bool isAdvertisingService(BLEUUID uuid); bool haveAppearance(); bool haveManufacturerData(); @@ -75,28 +115,15 @@ class BLEAdvertisedDevice { private: friend class BLEScan; - void parseAdvertisement(uint8_t *payload, size_t total_len = 62); - void setPayload(uint8_t *payload, size_t total_len = 62); - void setAddress(BLEAddress address); - void setAdFlag(uint8_t adFlag); - void setAdvertizementResult(uint8_t *payload); - void setAppearance(uint16_t appearance); - void setManufacturerData(String manufacturerData); - void setName(String name); - void setRSSI(int rssi); - void setScan(BLEScan *pScan); - void setServiceData(String data); - void setServiceDataUUID(BLEUUID uuid); - void setServiceUUID(const char *serviceUUID); - void setServiceUUID(BLEUUID serviceUUID); - void setTXPower(int8_t txPower); + /*************************************************************************** + * Common private properties * + ***************************************************************************/ bool m_haveAppearance; bool m_haveManufacturerData; bool m_haveName; bool m_haveRSSI; bool m_haveTXPower; - BLEAddress m_address = BLEAddress((uint8_t *)"\0\0\0\0\0\0"); uint8_t m_adFlag; uint16_t m_appearance; @@ -111,7 +138,37 @@ class BLEAdvertisedDevice { std::vector m_serviceDataUUIDs; uint8_t *m_payload; size_t m_payloadLength = 0; - esp_ble_addr_type_t m_addressType; + uint8_t m_addressType; + uint8_t m_advType; + bool m_isLegacyAdv; + + /*************************************************************************** + * NimBLE private properties * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) + bool m_callbackSent; +#endif + + /*************************************************************************** + * Common private declarations * + ***************************************************************************/ + + void parseAdvertisement(uint8_t *payload, size_t total_len = 62); + void setPayload(uint8_t *payload, size_t total_len = 62, bool append = false); + void setAddress(BLEAddress address); + void setAdFlag(uint8_t adFlag); + void setAdvertizementResult(uint8_t *payload); + void setAppearance(uint16_t appearance); + void setManufacturerData(String manufacturerData); + void setName(String name); + void setRSSI(int rssi); + void setScan(BLEScan *pScan); + void setServiceData(String data); + void setServiceDataUUID(BLEUUID uuid); + void setServiceUUID(const char *serviceUUID); + void setServiceUUID(BLEUUID serviceUUID); + void setTXPower(int8_t txPower); }; /** @@ -123,30 +180,55 @@ class BLEAdvertisedDevice { */ class BLEAdvertisedDeviceCallbacks { public: + /*************************************************************************** + * Common public declarations * + ***************************************************************************/ + virtual ~BLEAdvertisedDeviceCallbacks() {} /** - * @brief Called when a new scan result is detected. - * - * As we are scanning, we will find new devices. When found, this call back is invoked with a reference to the - * device that was found. During any individual scan, a device will only be detected one time. - */ + * @brief Called when a new scan result is detected. + * + * As we are scanning, we will find new devices. When found, this call back is invoked with a reference to the + * device that was found. During any individual scan, a device will only be detected one time. + */ virtual void onResult(BLEAdvertisedDevice advertisedDevice) = 0; }; -#ifdef SOC_BLE_50_SUPPORTED +#if defined(SOC_BLE_50_SUPPORTED) && defined(CONFIG_BLUEDROID_ENABLED) class BLEExtAdvertisingCallbacks { public: + /*************************************************************************** + * Common public declarations * + ***************************************************************************/ + virtual ~BLEExtAdvertisingCallbacks() {} + + /*************************************************************************** + * Bluedroid public declarations * + ***************************************************************************/ + /** - * @brief Called when a new scan result is detected. - * - * As we are scanning, we will find new devices. When found, this call back is invoked with a reference to the - * device that was found. During any individual scan, a device will only be detected one time. - */ + * @brief Called when a new scan result is detected. + * + * As we are scanning, we will find new devices. When found, this call back is invoked with a reference to the + * device that was found. During any individual scan, a device will only be detected one time. + */ + +#if defined(CONFIG_BLUEDROID_ENABLED) virtual void onResult(esp_ble_gap_ext_adv_report_t report) = 0; +#endif + + /*************************************************************************** + * NimBLE public declarations * + ***************************************************************************/ + + // Extended advertising for NimBLE is not supported yet. +#if defined(CONFIG_NIMBLE_ENABLED) + virtual void onResult(struct ble_gap_ext_disc_desc report) = 0; +#endif }; -#endif // SOC_BLE_50_SUPPORTED +#endif // SOC_BLE_50_SUPPORTED && CONFIG_BLUEDROID_ENABLED -#endif /* CONFIG_BLUEDROID_ENABLED */ +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ #endif /* COMPONENTS_CPP_UTILS_BLEADVERTISEDDEVICE_H_ */ diff --git a/libraries/BLE/src/BLEAdvertising.cpp b/libraries/BLE/src/BLEAdvertising.cpp index fe39a69c206..4e76873dcad 100644 --- a/libraries/BLE/src/BLEAdvertising.cpp +++ b/libraries/BLE/src/BLEAdvertising.cpp @@ -5,6 +5,10 @@ * Created on: Jun 21, 2017 * Author: kolban * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on kolban's and h2zero's work) + * Description: Added support for NimBLE + * * The ESP-IDF provides a framework for BLE advertising. It has determined that there are a common set * of properties that are advertised and has built a data structure that can be populated by the programmer. * This means that the programmer doesn't have to "mess with" the low level construction of a low level @@ -16,46 +20,33 @@ * set in the data will be advertised. * */ + #include "soc/soc_caps.h" #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) + +/*************************************************************************** + * Common includes * + ***************************************************************************/ + #include "BLEAdvertising.h" #include #include "BLEUtils.h" +#include "BLEDevice.h" #include "GeneralUtils.h" #include "esp32-hal-log.h" +/*************************************************************************** + * Common functions * + ***************************************************************************/ + /** * @brief Construct a default advertising object. - * */ -BLEAdvertising::BLEAdvertising() : m_scanRespData{} { - m_advData.set_scan_rsp = false; - m_advData.include_name = true; - m_advData.include_txpower = true; - m_advData.min_interval = 0x20; - m_advData.max_interval = 0x40; - m_advData.appearance = 0x00; - m_advData.manufacturer_len = 0; - m_advData.p_manufacturer_data = nullptr; - m_advData.service_data_len = 0; - m_advData.p_service_data = nullptr; - m_advData.service_uuid_len = 0; - m_advData.p_service_uuid = nullptr; - m_advData.flag = (ESP_BLE_ADV_FLAG_GEN_DISC | ESP_BLE_ADV_FLAG_BREDR_NOT_SPT); - - m_advParams.adv_int_min = 0x20; - m_advParams.adv_int_max = 0x40; - m_advParams.adv_type = ADV_TYPE_IND; - m_advParams.own_addr_type = BLE_ADDR_TYPE_PUBLIC; - m_advParams.channel_map = ADV_CHNL_ALL; - m_advParams.adv_filter_policy = ADV_FILTER_ALLOW_SCAN_ANY_CON_ANY; - m_advParams.peer_addr_type = BLE_ADDR_TYPE_PUBLIC; - - m_customAdvData = false; // No custom advertising data - m_customScanResponseData = false; // No custom scan response data +BLEAdvertising::BLEAdvertising() { + reset(); } // BLEAdvertising /** @@ -64,6 +55,9 @@ BLEAdvertising::BLEAdvertising() : m_scanRespData{} { */ void BLEAdvertising::addServiceUUID(BLEUUID serviceUUID) { m_serviceUUIDs.push_back(serviceUUID); +#ifdef CONFIG_NIMBLE_ENABLED + m_advDataSet = false; +#endif } // addServiceUUID /** @@ -72,6 +66,9 @@ void BLEAdvertising::addServiceUUID(BLEUUID serviceUUID) { */ void BLEAdvertising::addServiceUUID(const char *serviceUUID) { addServiceUUID(BLEUUID(serviceUUID)); +#ifdef CONFIG_NIMBLE_ENABLED + m_advDataSet = false; +#endif } // addServiceUUID /** @@ -87,6 +84,9 @@ bool BLEAdvertising::removeServiceUUID(int index) { } m_serviceUUIDs.erase(m_serviceUUIDs.begin() + index); +#ifdef CONFIG_NIMBLE_ENABLED + m_advDataSet = false; +#endif return true; } @@ -120,34 +120,107 @@ bool BLEAdvertising::removeServiceUUID(const char *serviceUUID) { */ void BLEAdvertising::setAppearance(uint16_t appearance) { m_advData.appearance = appearance; +#ifdef CONFIG_NIMBLE_ENABLED + m_advData.appearance_is_present = 1; + m_advDataSet = false; +#endif } // setAppearance -void BLEAdvertising::setAdvertisementType(esp_ble_adv_type_t adv_type) { - m_advParams.adv_type = adv_type; -} // setAdvertisementType +void BLEAdvertising::setAdvertisementType(uint8_t adv_type) { +#ifdef CONFIG_BLUEDROID_ENABLED + m_advParams.adv_type = (esp_ble_adv_type_t)adv_type; +#endif -void BLEAdvertising::setAdvertisementChannelMap(esp_ble_adv_channel_t channel_map) { - m_advParams.channel_map = channel_map; -} // setAdvertisementChannelMap +#if defined(CONFIG_NIMBLE_ENABLED) + m_advParams.conn_mode = adv_type; +#endif +} // setAdvertisementType void BLEAdvertising::setMinInterval(uint16_t mininterval) { +#ifdef CONFIG_BLUEDROID_ENABLED m_advParams.adv_int_min = mininterval; +#endif + +#if defined(CONFIG_NIMBLE_ENABLED) + m_advParams.itvl_min = mininterval; +#endif } // setMinInterval void BLEAdvertising::setMaxInterval(uint16_t maxinterval) { +#ifdef CONFIG_BLUEDROID_ENABLED m_advParams.adv_int_max = maxinterval; +#endif + +#if defined(CONFIG_NIMBLE_ENABLED) + m_advParams.itvl_max = maxinterval; +#endif } // setMaxInterval void BLEAdvertising::setMinPreferred(uint16_t mininterval) { +#ifdef CONFIG_BLUEDROID_ENABLED m_advData.min_interval = mininterval; +#endif + +#if defined(CONFIG_NIMBLE_ENABLED) + // invalid parameters, set the slave interval to null + if (mininterval < 0x0006 || mininterval > 0x0C80) { + m_advData.slave_itvl_range = nullptr; + return; + } + + if (m_advData.slave_itvl_range == nullptr) { + m_advData.slave_itvl_range = m_slaveItvl; + } + + m_slaveItvl[0] = mininterval; + m_slaveItvl[1] = mininterval >> 8; + + uint16_t maxinterval = *(uint16_t *)(m_advData.slave_itvl_range + 2); + + // If mininterval is higher than the maxinterval make them the same + if (mininterval > maxinterval) { + m_slaveItvl[2] = m_slaveItvl[0]; + m_slaveItvl[3] = m_slaveItvl[1]; + } + + m_advDataSet = false; +#endif } // void BLEAdvertising::setMaxPreferred(uint16_t maxinterval) { +#ifdef CONFIG_BLUEDROID_ENABLED m_advData.max_interval = maxinterval; +#endif + +#if defined(CONFIG_NIMBLE_ENABLED) + // invalid parameters, set the slave interval to null + if (maxinterval < 0x0006 || maxinterval > 0x0C80) { + m_advData.slave_itvl_range = nullptr; + return; + } + if (m_advData.slave_itvl_range == nullptr) { + m_advData.slave_itvl_range = m_slaveItvl; + } + m_slaveItvl[2] = maxinterval; + m_slaveItvl[3] = maxinterval >> 8; + + uint16_t mininterval = *(uint16_t *)(m_advData.slave_itvl_range); + + // If mininterval is higher than the maxinterval make them the same + if (mininterval > maxinterval) { + m_slaveItvl[0] = m_slaveItvl[2]; + m_slaveItvl[1] = m_slaveItvl[3]; + } + + m_advDataSet = false; +#endif } // void BLEAdvertising::setScanResponse(bool set) { m_scanResp = set; +#ifdef CONFIG_NIMBLE_ENABLED + m_advDataSet = false; +#endif } /** @@ -158,22 +231,54 @@ void BLEAdvertising::setScanResponse(bool set) { void BLEAdvertising::setScanFilter(bool scanRequestWhitelistOnly, bool connectWhitelistOnly) { log_v(">> setScanFilter: scanRequestWhitelistOnly: %d, connectWhitelistOnly: %d", scanRequestWhitelistOnly, connectWhitelistOnly); if (!scanRequestWhitelistOnly && !connectWhitelistOnly) { + +#ifdef CONFIG_BLUEDROID_ENABLED m_advParams.adv_filter_policy = ADV_FILTER_ALLOW_SCAN_ANY_CON_ANY; +#endif + +#if defined(CONFIG_NIMBLE_ENABLED) + m_advParams.filter_policy = BLE_HCI_ADV_FILT_NONE; +#endif + log_v("<< setScanFilter"); return; } if (scanRequestWhitelistOnly && !connectWhitelistOnly) { + +#ifdef CONFIG_BLUEDROID_ENABLED m_advParams.adv_filter_policy = ADV_FILTER_ALLOW_SCAN_WLST_CON_ANY; +#endif + +#if defined(CONFIG_NIMBLE_ENABLED) + m_advParams.filter_policy = BLE_HCI_ADV_FILT_SCAN; +#endif + log_v("<< setScanFilter"); return; } if (!scanRequestWhitelistOnly && connectWhitelistOnly) { + +#ifdef CONFIG_BLUEDROID_ENABLED m_advParams.adv_filter_policy = ADV_FILTER_ALLOW_SCAN_ANY_CON_WLST; +#endif + +#if defined(CONFIG_NIMBLE_ENABLED) + m_advParams.filter_policy = BLE_HCI_ADV_FILT_CONN; +#endif + log_v("<< setScanFilter"); return; } if (scanRequestWhitelistOnly && connectWhitelistOnly) { + +#ifdef CONFIG_BLUEDROID_ENABLED m_advParams.adv_filter_policy = ADV_FILTER_ALLOW_SCAN_WLST_CON_WLST; +#endif + +#if defined(CONFIG_NIMBLE_ENABLED) + m_advParams.filter_policy = BLE_HCI_ADV_FILT_BOTH; +#endif + log_v("<< setScanFilter"); return; } @@ -185,10 +290,21 @@ void BLEAdvertising::setScanFilter(bool scanRequestWhitelistOnly, bool connectWh */ bool BLEAdvertising::setAdvertisementData(BLEAdvertisementData &advertisementData) { log_v(">> setAdvertisementData"); + +#ifdef CONFIG_BLUEDROID_ENABLED esp_err_t errRc = ::esp_ble_gap_config_adv_data_raw((uint8_t *)advertisementData.getPayload().c_str(), advertisementData.getPayload().length()); if (errRc != ESP_OK) { log_e("esp_ble_gap_config_adv_data_raw: %d %s", errRc, GeneralUtils::errorToString(errRc)); } +#endif + +#if defined(CONFIG_NIMBLE_ENABLED) + esp_err_t errRc = ble_gap_adv_set_data((uint8_t *)advertisementData.getPayload().c_str(), advertisementData.getPayload().length()); + if (errRc != ESP_OK) { + log_e("ble_gap_adv_set_data: %d %s", errRc, BLEUtils::returnCodeToString(errRc)); + } +#endif + m_customAdvData = true; // Set the flag that indicates we are using custom advertising data. log_v("<< setAdvertisementData"); return ESP_OK == errRc; @@ -200,127 +316,25 @@ bool BLEAdvertising::setAdvertisementData(BLEAdvertisementData &advertisementDat */ bool BLEAdvertising::setScanResponseData(BLEAdvertisementData &advertisementData) { log_v(">> setScanResponseData"); + +#ifdef CONFIG_BLUEDROID_ENABLED esp_err_t errRc = ::esp_ble_gap_config_scan_rsp_data_raw((uint8_t *)advertisementData.getPayload().c_str(), advertisementData.getPayload().length()); if (errRc != ESP_OK) { log_e("esp_ble_gap_config_scan_rsp_data_raw: %d %s", errRc, GeneralUtils::errorToString(errRc)); } - m_customScanResponseData = true; // Set the flag that indicates we are using custom scan response data. - log_v("<< setScanResponseData"); - return ESP_OK == errRc; -} // setScanResponseData +#endif -/** - * @brief Start advertising. - * Start advertising. - * @return N/A. - */ -bool BLEAdvertising::start() { - log_v(">> start: customAdvData: %d, customScanResponseData: %d", m_customAdvData, m_customScanResponseData); - - // We have a vector of service UUIDs that we wish to advertise. In order to use the - // ESP-IDF framework, these must be supplied in a contiguous array of their 128bit (16 byte) - // representations. If we have 1 or more services to advertise then we allocate enough - // storage to host them and then copy them in one at a time into the contiguous storage. - int numServices = m_serviceUUIDs.size(); - if (numServices > 0) { - m_advData.service_uuid_len = 16 * numServices; - m_advData.p_service_uuid = (uint8_t *)malloc(m_advData.service_uuid_len); - if (!m_advData.p_service_uuid) { - log_e(">> start failed: out of memory"); - return false; - } - - uint8_t *p = m_advData.p_service_uuid; - for (int i = 0; i < numServices; i++) { - log_d("- advertising service: %s", m_serviceUUIDs[i].toString().c_str()); - BLEUUID serviceUUID128 = m_serviceUUIDs[i].to128(); - memcpy(p, serviceUUID128.getNative()->uuid.uuid128, 16); - p += 16; - } - } else { - m_advData.service_uuid_len = 0; - log_d("- no services advertised"); - } - - esp_err_t errRc; - - if (!m_customAdvData) { - // Set the configuration for advertising. - m_advData.set_scan_rsp = false; - m_advData.include_name = !m_scanResp; - m_advData.include_txpower = !m_scanResp; - errRc = ::esp_ble_gap_config_adv_data(&m_advData); - if (errRc != ESP_OK) { - log_e("<< esp_ble_gap_config_adv_data: rc=%d %s", errRc, GeneralUtils::errorToString(errRc)); - return false; - } - } - - if (!m_customScanResponseData && m_scanResp) { - // Set the configuration for scan response. - memcpy(&m_scanRespData, &m_advData, sizeof(esp_ble_adv_data_t)); // Copy the content of m_advData. - m_scanRespData.set_scan_rsp = true; // Define this struct as scan response data - m_scanRespData.include_name = true; // Caution: This may lead to a crash if the device name has more than 29 characters - m_scanRespData.include_txpower = true; - m_scanRespData.appearance = 0; // If defined the 'Appearance' attribute is already included in the advertising data - m_scanRespData.flag = 0; // 'Flags' attribute should no be included in the scan response - - errRc = ::esp_ble_gap_config_adv_data(&m_scanRespData); - if (errRc != ESP_OK) { - log_e("<< esp_ble_gap_config_adv_data (Scan response): rc=%d %s", errRc, GeneralUtils::errorToString(errRc)); - return false; - } - } - - // If we had services to advertise then we previously allocated some storage for them. - // Here we release that storage. - free(m_advData.p_service_uuid); //TODO change this variable to local scope? - m_advData.p_service_uuid = nullptr; - - // Start advertising. - errRc = ::esp_ble_gap_start_advertising(&m_advParams); +#if defined(CONFIG_NIMBLE_ENABLED) + esp_err_t errRc = ble_gap_adv_rsp_set_data((uint8_t *)advertisementData.getPayload().c_str(), advertisementData.getPayload().length()); if (errRc != ESP_OK) { - log_e("<< esp_ble_gap_start_advertising: rc=%d %s", errRc, GeneralUtils::errorToString(errRc)); - } else { - log_v("<< start"); + log_e("ble_gap_adv_rsp_set_data: %d %s", errRc, BLEUtils::returnCodeToString(errRc)); } - return ESP_OK == errRc; -} // start - -/** - * @brief Stop advertising. - * Stop advertising. - * @return N/A. - */ -bool BLEAdvertising::stop() { - log_v(">> stop"); - esp_err_t errRc = ::esp_ble_gap_stop_advertising(); - if (errRc != ESP_OK) { - log_e("esp_ble_gap_stop_advertising: rc=%d %s", errRc, GeneralUtils::errorToString(errRc)); - } else { - log_v("<< stop"); - } - return ESP_OK == errRc; -} // stop - -/** - * @brief Set BLE address. - * @param [in] Bluetooth address. - * @param [in] Bluetooth address type. - * Set BLE address. - */ -bool BLEAdvertising::setDeviceAddress(esp_bd_addr_t addr, esp_ble_addr_type_t type) { - log_v(">> setPrivateAddress"); +#endif - m_advParams.own_addr_type = type; - esp_err_t errRc = esp_ble_gap_set_rand_addr((uint8_t *)addr); - if (errRc != ESP_OK) { - log_e("esp_ble_gap_set_rand_addr: rc=%d %s", errRc, GeneralUtils::errorToString(errRc)); - } else { - log_v("<< setPrivateAddress"); - } + m_customScanResponseData = true; // Set the flag that indicates we are using custom scan response data. + log_v("<< setScanResponseData"); return ESP_OK == errRc; -} // setPrivateAddress +} // setScanResponseData /** * @brief Add data to the payload to be advertised. @@ -333,6 +347,13 @@ void BLEAdvertisementData::addData(String data) { m_payload.concat(data); } // addData +void BLEAdvertisementData::addData(char *data, size_t length) { + if ((m_payload.length() + length) > ESP_BLE_ADV_DATA_LEN_MAX) { + return; + } + m_payload.concat(String(data, length)); +} // addData + /** * @brief Set the appearance. * @param [in] appearance The appearance code value. @@ -359,7 +380,12 @@ void BLEAdvertisementData::setCompleteServices(BLEUUID uuid) { // [Len] [0x03] [LL] [HH] cdata[0] = 3; cdata[1] = ESP_BLE_AD_TYPE_16SRV_CMPL; // 0x03 +#if defined(CONFIG_BLUEDROID_ENABLED) addData(String(cdata, 2) + String((char *)&uuid.getNative()->uuid.uuid16, 2)); +#endif +#if defined(CONFIG_NIMBLE_ENABLED) + addData(String(cdata, 2) + String((char *)&uuid.getNative()->u16.value, 2)); +#endif break; } @@ -368,7 +394,12 @@ void BLEAdvertisementData::setCompleteServices(BLEUUID uuid) { // [Len] [0x05] [LL] [LL] [HH] [HH] cdata[0] = 5; cdata[1] = ESP_BLE_AD_TYPE_32SRV_CMPL; // 0x05 +#if defined(CONFIG_BLUEDROID_ENABLED) addData(String(cdata, 2) + String((char *)&uuid.getNative()->uuid.uuid32, 4)); +#endif +#if defined(CONFIG_NIMBLE_ENABLED) + addData(String(cdata, 2) + String((char *)&uuid.getNative()->u32.value, 4)); +#endif break; } @@ -377,7 +408,12 @@ void BLEAdvertisementData::setCompleteServices(BLEUUID uuid) { // [Len] [0x07] [0] [1] ... [15] cdata[0] = 17; cdata[1] = ESP_BLE_AD_TYPE_128SRV_CMPL; // 0x07 +#if defined(CONFIG_BLUEDROID_ENABLED) addData(String(cdata, 2) + String((char *)uuid.getNative()->uuid.uuid128, 16)); +#endif +#if defined(CONFIG_NIMBLE_ENABLED) + addData(String(cdata, 2) + String((char *)uuid.getNative()->u128.value, 16)); +#endif break; } @@ -442,7 +478,12 @@ void BLEAdvertisementData::setPartialServices(BLEUUID uuid) { // [Len] [0x02] [LL] [HH] cdata[0] = 3; cdata[1] = ESP_BLE_AD_TYPE_16SRV_PART; // 0x02 +#if defined(CONFIG_BLUEDROID_ENABLED) addData(String(cdata, 2) + String((char *)&uuid.getNative()->uuid.uuid16, 2)); +#endif +#if defined(CONFIG_NIMBLE_ENABLED) + addData(String(cdata, 2) + String((char *)&uuid.getNative()->u16.value, 2)); +#endif break; } @@ -451,7 +492,12 @@ void BLEAdvertisementData::setPartialServices(BLEUUID uuid) { // [Len] [0x04] [LL] [LL] [HH] [HH] cdata[0] = 5; cdata[1] = ESP_BLE_AD_TYPE_32SRV_PART; // 0x04 +#if defined(CONFIG_BLUEDROID_ENABLED) addData(String(cdata, 2) + String((char *)&uuid.getNative()->uuid.uuid32, 4)); +#endif +#if defined(CONFIG_NIMBLE_ENABLED) + addData(String(cdata, 2) + String((char *)&uuid.getNative()->u32.value, 4)); +#endif break; } @@ -460,7 +506,12 @@ void BLEAdvertisementData::setPartialServices(BLEUUID uuid) { // [Len] [0x06] [0] [1] ... [15] cdata[0] = 17; cdata[1] = ESP_BLE_AD_TYPE_128SRV_PART; // 0x06 +#if defined(CONFIG_BLUEDROID_ENABLED) addData(String(cdata, 2) + String((char *)&uuid.getNative()->uuid.uuid128, 16)); +#endif +#if defined(CONFIG_NIMBLE_ENABLED) + addData(String(cdata, 2) + String((char *)&uuid.getNative()->u128.value, 16)); +#endif break; } @@ -481,7 +532,12 @@ void BLEAdvertisementData::setServiceData(BLEUUID uuid, String data) { // [Len] [0x16] [UUID16] data cdata[0] = data.length() + 3; cdata[1] = ESP_BLE_AD_TYPE_SERVICE_DATA; // 0x16 +#if defined(CONFIG_BLUEDROID_ENABLED) addData(String(cdata, 2) + String((char *)&uuid.getNative()->uuid.uuid16, 2) + data); +#endif +#if defined(CONFIG_NIMBLE_ENABLED) + addData(String(cdata, 2) + String((char *)&uuid.getNative()->u16.value, 2) + data); +#endif break; } @@ -490,7 +546,12 @@ void BLEAdvertisementData::setServiceData(BLEUUID uuid, String data) { // [Len] [0x20] [UUID32] data cdata[0] = data.length() + 5; cdata[1] = ESP_BLE_AD_TYPE_32SERVICE_DATA; // 0x20 +#if defined(CONFIG_BLUEDROID_ENABLED) addData(String(cdata, 2) + String((char *)&uuid.getNative()->uuid.uuid32, 4) + data); +#endif +#if defined(CONFIG_NIMBLE_ENABLED) + addData(String(cdata, 2) + String((char *)&uuid.getNative()->u32.value, 4) + data); +#endif break; } @@ -499,7 +560,12 @@ void BLEAdvertisementData::setServiceData(BLEUUID uuid, String data) { // [Len] [0x21] [UUID128] data cdata[0] = data.length() + 17; cdata[1] = ESP_BLE_AD_TYPE_128SERVICE_DATA; // 0x21 +#if defined(CONFIG_BLUEDROID_ENABLED) addData(String(cdata, 2) + String((char *)&uuid.getNative()->uuid.uuid128, 16) + data); +#endif +#if defined(CONFIG_NIMBLE_ENABLED) + addData(String(cdata, 2) + String((char *)&uuid.getNative()->u128.value, 16) + data); +#endif break; } @@ -520,6 +586,33 @@ void BLEAdvertisementData::setShortName(String name) { log_d("BLEAdvertisementData", "<< setShortName"); } // setShortName +/** + * @brief Adds Tx power level to the advertisement data. + */ +void BLEAdvertisementData::addTxPower() { + char cdata[3]; + cdata[0] = 2; // length + cdata[1] = ESP_BLE_AD_TYPE_TX_PWR; + cdata[2] = BLEDevice::getPower(); + addData(cdata, 3); +} // addTxPower + +/** + * @brief Set the preferred connection interval parameters. + * @param [in] min The minimum interval desired. + * @param [in] max The maximum interval desired. + */ +void BLEAdvertisementData::setPreferredParams(uint16_t min, uint16_t max) { + char cdata[6]; + cdata[0] = 5; // length + cdata[1] = ESP_BLE_AD_TYPE_INT_RANGE; + cdata[2] = min; + cdata[3] = min >> 8; + cdata[4] = max; + cdata[5] = max >> 8; + addData(cdata, 6); +} // setPreferredParams + /** * @brief Retrieve the payload that is to be advertised. * @return The payload that is to be advertised. @@ -528,8 +621,146 @@ String BLEAdvertisementData::getPayload() { return m_payload; } // getPayload -void BLEAdvertising::handleGAPEvent(esp_gap_ble_cb_event_t event, esp_ble_gap_cb_param_t *param) { +/*************************************************************************** + * Bluedroid functions * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) + +void BLEAdvertising::reset() { + if (BLEDevice::getInitialized()) { + stop(); + } + + memset(&m_scanRespData, 0, sizeof(esp_ble_adv_data_t)); + memset(&m_advData, 0, sizeof(esp_ble_adv_data_t)); + memset(&m_advParams, 0, sizeof(esp_ble_adv_params_t)); + + m_advData.set_scan_rsp = false; + m_advData.include_name = true; + m_advData.include_txpower = true; + m_advData.min_interval = 0x20; + m_advData.max_interval = 0x40; + m_advData.appearance = 0x00; + m_advData.manufacturer_len = 0; + m_advData.p_manufacturer_data = nullptr; + m_advData.service_data_len = 0; + m_advData.p_service_data = nullptr; + m_advData.service_uuid_len = 0; + m_advData.p_service_uuid = nullptr; + m_advData.flag = (ESP_BLE_ADV_FLAG_GEN_DISC | ESP_BLE_ADV_FLAG_BREDR_NOT_SPT); + + m_advParams.adv_int_min = 0x20; + m_advParams.adv_int_max = 0x40; + m_advParams.adv_type = ADV_TYPE_IND; + m_advParams.own_addr_type = BLE_ADDR_TYPE_PUBLIC; + m_advParams.channel_map = ADV_CHNL_ALL; + m_advParams.adv_filter_policy = ADV_FILTER_ALLOW_SCAN_ANY_CON_ANY; + m_advParams.peer_addr_type = BLE_ADDR_TYPE_PUBLIC; + m_customAdvData = false; // No custom advertising data + m_customScanResponseData = false; // No custom scan response data +} // BLEAdvertising + +void BLEAdvertising::setAdvertisementChannelMap(esp_ble_adv_channel_t channel_map) { + m_advParams.channel_map = channel_map; +} // setAdvertisementChannelMap + +/** + * @brief Start advertising. + * Start advertising. + * @return N/A. + */ +bool BLEAdvertising::start() { + log_v(">> start: customAdvData: %d, customScanResponseData: %d", m_customAdvData, m_customScanResponseData); + + // We have a vector of service UUIDs that we wish to advertise. In order to use the + // ESP-IDF framework, these must be supplied in a contiguous array of their 128bit (16 byte) + // representations. If we have 1 or more services to advertise then we allocate enough + // storage to host them and then copy them in one at a time into the contiguous storage. + int numServices = m_serviceUUIDs.size(); + if (numServices > 0) { + m_advData.service_uuid_len = 16 * numServices; + m_advData.p_service_uuid = (uint8_t *)malloc(m_advData.service_uuid_len); + if (!m_advData.p_service_uuid) { + log_e(">> start failed: out of memory"); + return false; + } + + uint8_t *p = m_advData.p_service_uuid; + for (int i = 0; i < numServices; i++) { + log_d("- advertising service: %s", m_serviceUUIDs[i].toString().c_str()); + BLEUUID serviceUUID128 = m_serviceUUIDs[i].to128(); + memcpy(p, serviceUUID128.getNative()->uuid.uuid128, 16); + p += 16; + } + } else { + m_advData.service_uuid_len = 0; + log_d("- no services advertised"); + } + + esp_err_t errRc; + + if (!m_customAdvData) { + // Set the configuration for advertising. + m_advData.set_scan_rsp = false; + m_advData.include_name = !m_scanResp; + m_advData.include_txpower = !m_scanResp; + errRc = ::esp_ble_gap_config_adv_data(&m_advData); + if (errRc != ESP_OK) { + log_e("<< esp_ble_gap_config_adv_data: rc=%d %s", errRc, GeneralUtils::errorToString(errRc)); + return false; + } + } + + if (!m_customScanResponseData && m_scanResp) { + // Set the configuration for scan response. + memcpy(&m_scanRespData, &m_advData, sizeof(esp_ble_adv_data_t)); // Copy the content of m_advData. + m_scanRespData.set_scan_rsp = true; // Define this struct as scan response data + m_scanRespData.include_name = true; // Caution: This may lead to a crash if the device name has more than 29 characters + m_scanRespData.include_txpower = true; + m_scanRespData.appearance = 0; // If defined the 'Appearance' attribute is already included in the advertising data + m_scanRespData.flag = 0; // 'Flags' attribute should no be included in the scan response + + errRc = ::esp_ble_gap_config_adv_data(&m_scanRespData); + if (errRc != ESP_OK) { + log_e("<< esp_ble_gap_config_adv_data (Scan response): rc=%d %s", errRc, GeneralUtils::errorToString(errRc)); + return false; + } + } + + // If we had services to advertise then we previously allocated some storage for them. + // Here we release that storage. + free(m_advData.p_service_uuid); //TODO change this variable to local scope? + m_advData.p_service_uuid = nullptr; + + // Start advertising. + errRc = ::esp_ble_gap_start_advertising(&m_advParams); + if (errRc != ESP_OK) { + log_e("<< esp_ble_gap_start_advertising: rc=%d %s", errRc, GeneralUtils::errorToString(errRc)); + } else { + log_v("<< start"); + } + return ESP_OK == errRc; +} // start + +/** + * @brief Stop advertising. + * Stop advertising. + * @return N/A. + */ +bool BLEAdvertising::stop() { + log_v(">> stop"); + esp_err_t errRc = ::esp_ble_gap_stop_advertising(); + if (errRc != ESP_OK) { + log_e("esp_ble_gap_stop_advertising: rc=%d %s", errRc, GeneralUtils::errorToString(errRc)); + } else { + log_v("<< stop"); + } + return ESP_OK == errRc; +} // stop + +void BLEAdvertising::handleGAPEvent(esp_gap_ble_cb_event_t event, esp_ble_gap_cb_param_t *param) { log_d("handleGAPEvent [event no: %d]", (int)event); switch (event) { @@ -558,7 +789,26 @@ void BLEAdvertising::handleGAPEvent(esp_gap_ble_cb_event_t event, esp_ble_gap_cb } } -#ifdef SOC_BLE_50_SUPPORTED +/** + * @brief Set BLE address. + * @param [in] Bluetooth address. + * @param [in] Bluetooth address type. + * Set BLE address. + */ +bool BLEAdvertising::setDeviceAddress(esp_bd_addr_t addr, esp_ble_addr_type_t type) { + log_v(">> setPrivateAddress"); + + m_advParams.own_addr_type = type; + esp_err_t errRc = esp_ble_gap_set_rand_addr((uint8_t *)addr); + if (errRc != ESP_OK) { + log_e("esp_ble_gap_set_rand_addr: rc=%d %s", errRc, GeneralUtils::errorToString(errRc)); + } else { + log_v("<< setPrivateAddress"); + } + return ESP_OK == errRc; +} // setPrivateAddress + +#if defined(SOC_BLE_50_SUPPORTED) /** * @brief Creator @@ -803,7 +1053,376 @@ void BLEMultiAdvertising::setDuration(uint8_t instance, int duration, int max_ev ext_adv[instance] = {instance, duration, max_events}; } -#endif // SOC_BLE_50_SUPPORTED +#endif /* SOC_BLE_50_SUPPORTED */ #endif /* CONFIG_BLUEDROID_ENABLED */ + +/*************************************************************************** + * NimBLE functions * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) + +void BLEAdvertising::reset() { + if (BLEDevice::getInitialized() && isAdvertising()) { + stop(); + } + memset(&m_advData, 0, sizeof m_advData); + memset(&m_scanData, 0, sizeof m_scanData); + memset(&m_advParams, 0, sizeof m_advParams); + memset(&m_slaveItvl, 0, sizeof m_slaveItvl); + const char *name = ble_svc_gap_device_name(); + + m_advData.name = (uint8_t *)name; + m_advData.name_len = strlen(name); + m_advData.name_is_complete = 1; + m_advData.tx_pwr_lvl = BLEDevice::getPower(); + m_advData.flags = (BLE_HS_ADV_F_DISC_GEN | BLE_HS_ADV_F_BREDR_UNSUP); + m_advParams.conn_mode = BLE_GAP_CONN_MODE_UND; + m_advParams.disc_mode = BLE_GAP_DISC_MODE_GEN; + m_customAdvData = false; + m_customScanResponseData = false; + m_scanResp = true; + m_advDataSet = false; + // Set this to non-zero to prevent auto start if host reset before started by app. + m_duration = BLE_HS_FOREVER; + m_advCompCB = nullptr; +} // BLEAdvertising + +void BLEAdvertising::setName(String name) { + m_name = name; + m_advData.name = (uint8_t *)m_name.c_str(); + m_advData.name_len = m_name.length(); + m_advDataSet = false; +} + +/** + * @brief Add the transmission power level to the advertisement packet. + */ +void BLEAdvertising::addTxPower() { + m_advData.tx_pwr_lvl_is_present = 1; + m_advDataSet = false; +} // addTxPower + +/** + * @brief Handles the callback when advertising stops. + */ +void BLEAdvertising::advCompleteCB() { + if (m_advCompCB != nullptr) { + m_advCompCB(this); + } +} // advCompleteCB + +/** + * @brief Check if currently advertising. + * @return true if advertising is active. + */ +bool BLEAdvertising::isAdvertising() { + return ble_gap_adv_active(); +} // isAdvertising + +/* + * Host reset seems to clear advertising data, + * we need clear the flag so it reloads it. + */ +void BLEAdvertising::onHostSync() { + log_v("Host re-synced"); + + m_advDataSet = false; + // If we were advertising forever, restart it now + if (m_duration == 0) { + start(m_duration, m_advCompCB); + } else { + // Otherwise we should tell the app that advertising stopped. + advCompleteCB(); + } +} // onHostSync + +/** + * @brief Handler for gap events when not using peripheral role. + * @param [in] event the event data. + * @param [in] arg pointer to the advertising instance. + */ +int BLEAdvertising::handleGAPEvent(struct ble_gap_event *event, void *arg) { + BLEAdvertising *pAdv = (BLEAdvertising *)arg; + + if (event->type == BLE_GAP_EVENT_ADV_COMPLETE) { + switch (event->adv_complete.reason) { + // Don't call the callback if host reset, we want to + // preserve the active flag until re-sync to restart advertising. + case BLE_HS_ETIMEOUT_HCI: + case BLE_HS_EOS: + case BLE_HS_ECONTROLLER: + case BLE_HS_ENOTSYNCED: + log_e("host reset, rc=%d", event->adv_complete.reason); + BLEDevice::onReset(event->adv_complete.reason); + return 0; + default: break; + } + pAdv->advCompleteCB(); + } + return 0; +} + +/** + * @brief Start advertising. + * @param [in] duration The duration, in seconds, to advertise, 0 == advertise forever. + * @param [in] advCompleteCB A pointer to a callback to be invoked when advertising ends. + * @return True if advertising started successfully. + */ +bool BLEAdvertising::start(uint32_t duration, void (*advCompleteCB)(BLEAdvertising *pAdv)) { + log_v(">> Advertising start: customAdvData: %d, customScanResponseData: %d", m_customAdvData, m_customScanResponseData); + + // If Host is not synced we cannot start advertising. + if (!BLEDevice::m_synced) { + log_e("Host reset, wait for sync."); + return false; + } + + // If already advertising just return + if (ble_gap_adv_active()) { + log_w("Advertising already active"); + return true; + } + + BLEServer *pServer = BLEDevice::getServer(); + if (pServer != nullptr) { + if (!pServer->m_gattsStarted) { + pServer->start(); + } else if (pServer->getConnectedCount() >= CONFIG_BT_NIMBLE_MAX_CONNECTIONS) { + log_e("Max connections reached - not advertising"); + return false; + } + } + + // Save the duration in case of host reset so we can restart with the same parameters + m_duration = duration; + + if (duration == 0) { + duration = BLE_HS_FOREVER; + } else { + duration = duration * 1000; // convert duration to milliseconds + } + + m_advCompCB = advCompleteCB; + + m_advParams.disc_mode = BLE_GAP_DISC_MODE_GEN; + m_advData.flags = (BLE_HS_ADV_F_DISC_GEN | BLE_HS_ADV_F_BREDR_UNSUP); + if (m_advParams.conn_mode == BLE_GAP_CONN_MODE_NON) { + if (!m_scanResp) { + m_advParams.disc_mode = BLE_GAP_DISC_MODE_NON; + m_advData.flags = BLE_HS_ADV_F_BREDR_UNSUP; + } + } + + int rc = 0; + + if (!m_customAdvData && !m_advDataSet) { + //start with 3 bytes for the flags data + uint8_t payloadLen = (2 + 1); + if (m_advData.mfg_data_len > 0) { + payloadLen += (2 + m_advData.mfg_data_len); + } + + if (m_advData.svc_data_uuid16_len > 0) { + payloadLen += (2 + m_advData.svc_data_uuid16_len); + } + + if (m_advData.svc_data_uuid32_len > 0) { + payloadLen += (2 + m_advData.svc_data_uuid32_len); + } + + if (m_advData.svc_data_uuid128_len > 0) { + payloadLen += (2 + m_advData.svc_data_uuid128_len); + } + + if (m_advData.uri_len > 0) { + payloadLen += (2 + m_advData.uri_len); + } + + if (m_advData.appearance_is_present) { + payloadLen += (2 + BLE_HS_ADV_APPEARANCE_LEN); + } + + if (m_advData.tx_pwr_lvl_is_present) { + payloadLen += (2 + BLE_HS_ADV_TX_PWR_LVL_LEN); + } + + if (m_advData.slave_itvl_range != nullptr) { + payloadLen += (2 + BLE_HS_ADV_SLAVE_ITVL_RANGE_LEN); + } + + for (auto &it : m_serviceUUIDs) { + if (it.getNative()->u.type == BLE_UUID_TYPE_16) { + int add = (m_advData.num_uuids16 > 0) ? 2 : 4; + if ((payloadLen + add) > BLE_HS_ADV_MAX_SZ) { + m_advData.uuids16_is_complete = 0; + continue; + } + payloadLen += add; + + if (nullptr == (m_advData.uuids16 = (ble_uuid16_t *)realloc((void *)m_advData.uuids16, (m_advData.num_uuids16 + 1) * sizeof(ble_uuid16_t)))) { + log_e("Error, no mem"); + abort(); + } + memcpy((void *)&m_advData.uuids16[m_advData.num_uuids16], &it.getNative()->u16, sizeof(ble_uuid16_t)); + m_advData.uuids16_is_complete = 1; + m_advData.num_uuids16++; + } + if (it.getNative()->u.type == BLE_UUID_TYPE_32) { + int add = (m_advData.num_uuids32 > 0) ? 4 : 6; + if ((payloadLen + add) > BLE_HS_ADV_MAX_SZ) { + m_advData.uuids32_is_complete = 0; + continue; + } + payloadLen += add; + + if (nullptr == (m_advData.uuids32 = (ble_uuid32_t *)realloc((void *)m_advData.uuids32, (m_advData.num_uuids32 + 1) * sizeof(ble_uuid32_t)))) { + log_e("Error, no mem"); + abort(); + } + memcpy((void *)&m_advData.uuids32[m_advData.num_uuids32], &it.getNative()->u32, sizeof(ble_uuid32_t)); + m_advData.uuids32_is_complete = 1; + m_advData.num_uuids32++; + } + if (it.getNative()->u.type == BLE_UUID_TYPE_128) { + int add = (m_advData.num_uuids128 > 0) ? 16 : 18; + if ((payloadLen + add) > BLE_HS_ADV_MAX_SZ) { + m_advData.uuids128_is_complete = 0; + continue; + } + payloadLen += add; + + if (nullptr == (m_advData.uuids128 = (ble_uuid128_t *)realloc((void *)m_advData.uuids128, (m_advData.num_uuids128 + 1) * sizeof(ble_uuid128_t)))) { + log_e("Error, no mem"); + abort(); + } + memcpy((void *)&m_advData.uuids128[m_advData.num_uuids128], &it.getNative()->u128, sizeof(ble_uuid128_t)); + m_advData.uuids128_is_complete = 1; + m_advData.num_uuids128++; + } + } + + // check if there is room for the name, if not put it in scan data + if ((payloadLen + (2 + m_advData.name_len)) > BLE_HS_ADV_MAX_SZ) { + if (m_scanResp && !m_customScanResponseData) { + m_scanData.name = m_advData.name; + m_scanData.name_len = m_advData.name_len; + if (m_scanData.name_len > BLE_HS_ADV_MAX_SZ - 2) { + m_scanData.name_len = BLE_HS_ADV_MAX_SZ - 2; + m_scanData.name_is_complete = 0; + } else { + m_scanData.name_is_complete = 1; + } + m_advData.name = nullptr; + m_advData.name_len = 0; + m_advData.name_is_complete = 0; + } else { + if (m_advData.tx_pwr_lvl_is_present) { + m_advData.tx_pwr_lvl_is_present = 0; + payloadLen -= (2 + 1); + } + // if not using scan response just cut the name down + // leaving 2 bytes for the data specifier. + if (m_advData.name_len > (BLE_HS_ADV_MAX_SZ - payloadLen - 2)) { + m_advData.name_len = (BLE_HS_ADV_MAX_SZ - payloadLen - 2); + m_advData.name_is_complete = 0; + } + } + } + + if (m_scanResp && !m_customScanResponseData) { + rc = ble_gap_adv_rsp_set_fields(&m_scanData); + switch (rc) { + case 0: break; + + case BLE_HS_EBUSY: log_e("Already advertising"); break; + + case BLE_HS_EMSGSIZE: log_e("Scan data too long"); break; + + default: log_e("Error setting scan response data; rc=%d, %s", rc, BLEUtils::returnCodeToString(rc)); break; + } + } + + if (rc == 0) { + rc = ble_gap_adv_set_fields(&m_advData); + switch (rc) { + case 0: break; + + case BLE_HS_EBUSY: log_e("Already advertising"); break; + + case BLE_HS_EMSGSIZE: log_e("Advertisement data too long"); break; + + default: log_e("Error setting advertisement data; rc=%d, %s", rc, BLEUtils::returnCodeToString(rc)); break; + } + } + + if (m_advData.num_uuids128 > 0) { + free((void *)m_advData.uuids128); + m_advData.uuids128 = nullptr; + m_advData.num_uuids128 = 0; + } + + if (m_advData.num_uuids32 > 0) { + free((void *)m_advData.uuids32); + m_advData.uuids32 = nullptr; + m_advData.num_uuids32 = 0; + } + + if (m_advData.num_uuids16 > 0) { + free((void *)m_advData.uuids16); + m_advData.uuids16 = nullptr; + m_advData.num_uuids16 = 0; + } + + if (rc != 0) { + return false; + } + + m_advDataSet = true; + } + + rc = ble_gap_adv_start( + BLEDevice::m_ownAddrType, NULL, duration, &m_advParams, (pServer != nullptr) ? BLEServer::handleGATTServerEvent : BLEAdvertising::handleGAPEvent, + (pServer != nullptr) ? (void *)pServer : (void *)this + ); + + switch (rc) { + case 0: break; + + case BLE_HS_EINVAL: log_e("Unable to advertise - Duration too long"); break; + + case BLE_HS_EPREEMPTED: log_e("Unable to advertise - busy"); break; + + case BLE_HS_ETIMEOUT_HCI: + case BLE_HS_EOS: + case BLE_HS_ECONTROLLER: + case BLE_HS_ENOTSYNCED: log_e("Unable to advertise - Host Reset"); break; + + default: log_e("Error enabling advertising; rc=%d, %s", rc, BLEUtils::returnCodeToString(rc)); break; + } + + log_d("<< Advertising start"); + return (rc == 0); +} // start + +/** + * @brief Stop advertising. + */ +bool BLEAdvertising::stop() { + log_d(">> stop"); + + int rc = ble_gap_adv_stop(); + if (rc != 0 && rc != BLE_HS_EALREADY) { + log_e("ble_gap_adv_stop rc=%d %s", rc, BLEUtils::returnCodeToString(rc)); + return false; + } + + log_d("<< stop"); + return true; +} // stop + +#endif /* CONFIG_NIMBLE_ENABLED */ + +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ diff --git a/libraries/BLE/src/BLEAdvertising.h b/libraries/BLE/src/BLEAdvertising.h index 1e573ac814f..cf68768ec91 100644 --- a/libraries/BLE/src/BLEAdvertising.h +++ b/libraries/BLE/src/BLEAdvertising.h @@ -3,6 +3,10 @@ * * Created on: Jun 21, 2017 * Author: kolban + * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on kolban's and h2zero's work) + * Description: Added support for NimBLE */ #ifndef COMPONENTS_CPP_UTILS_BLEADVERTISING_H_ @@ -11,20 +15,97 @@ #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) -#include +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) + +/*************************************************************************** + * Common includes * + ***************************************************************************/ + #include "BLEUUID.h" #include #include "RTOS.h" +#include "BLEUtils.h" + +/*************************************************************************** + * Bluedroid includes * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) +#include +#endif + +/*************************************************************************** + * NimBLE includes and definitions * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) +#include +#include +#include + +#define ESP_BLE_ADV_DATA_LEN_MAX BLE_HS_ADV_MAX_SZ +#define ESP_BLE_ADV_FLAG_LIMIT_DISC (0x01 << 0) +#define ESP_BLE_ADV_FLAG_GEN_DISC (0x01 << 1) +#define ESP_BLE_ADV_FLAG_BREDR_NOT_SPT (0x01 << 2) +#define ESP_BLE_ADV_FLAG_DMT_CONTROLLER_SPT (0x01 << 3) +#define ESP_BLE_ADV_FLAG_DMT_HOST_SPT (0x01 << 4) +#define ESP_BLE_ADV_FLAG_NON_LIMIT_DISC (0x00) +#endif /* CONFIG_NIMBLE_ENABLED */ + +/*************************************************************************** + * NimBLE types * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) +typedef enum { + ESP_BLE_AD_TYPE_FLAG = 0x01, + ESP_BLE_AD_TYPE_16SRV_PART = 0x02, + ESP_BLE_AD_TYPE_16SRV_CMPL = 0x03, + ESP_BLE_AD_TYPE_32SRV_PART = 0x04, + ESP_BLE_AD_TYPE_32SRV_CMPL = 0x05, + ESP_BLE_AD_TYPE_128SRV_PART = 0x06, + ESP_BLE_AD_TYPE_128SRV_CMPL = 0x07, + ESP_BLE_AD_TYPE_NAME_SHORT = 0x08, + ESP_BLE_AD_TYPE_NAME_CMPL = 0x09, + ESP_BLE_AD_TYPE_TX_PWR = 0x0A, + ESP_BLE_AD_TYPE_DEV_CLASS = 0x0D, + ESP_BLE_AD_TYPE_SM_TK = 0x10, + ESP_BLE_AD_TYPE_SM_OOB_FLAG = 0x11, + ESP_BLE_AD_TYPE_INT_RANGE = 0x12, + ESP_BLE_AD_TYPE_SOL_SRV_UUID = 0x14, + ESP_BLE_AD_TYPE_128SOL_SRV_UUID = 0x15, + ESP_BLE_AD_TYPE_SERVICE_DATA = 0x16, + ESP_BLE_AD_TYPE_PUBLIC_TARGET = 0x17, + ESP_BLE_AD_TYPE_RANDOM_TARGET = 0x18, + ESP_BLE_AD_TYPE_APPEARANCE = 0x19, + ESP_BLE_AD_TYPE_ADV_INT = 0x1A, + ESP_BLE_AD_TYPE_LE_DEV_ADDR = 0x1b, + ESP_BLE_AD_TYPE_LE_ROLE = 0x1c, + ESP_BLE_AD_TYPE_SPAIR_C256 = 0x1d, + ESP_BLE_AD_TYPE_SPAIR_R256 = 0x1e, + ESP_BLE_AD_TYPE_32SOL_SRV_UUID = 0x1f, + ESP_BLE_AD_TYPE_32SERVICE_DATA = 0x20, + ESP_BLE_AD_TYPE_128SERVICE_DATA = 0x21, + ESP_BLE_AD_TYPE_LE_SECURE_CONFIRM = 0x22, + ESP_BLE_AD_TYPE_LE_SECURE_RANDOM = 0x23, + ESP_BLE_AD_TYPE_URI = 0x24, + ESP_BLE_AD_TYPE_INDOOR_POSITION = 0x25, + ESP_BLE_AD_TYPE_TRANS_DISC_DATA = 0x26, + ESP_BLE_AD_TYPE_LE_SUPPORT_FEATURE = 0x27, + ESP_BLE_AD_TYPE_CHAN_MAP_UPDATE = 0x28, + ESP_BLE_AD_MANUFACTURER_SPECIFIC_TYPE = 0xFF, +} esp_ble_adv_data_type; +#endif /** * @brief Advertisement data set by the programmer to be published by the %BLE server. */ class BLEAdvertisementData { - // Only a subset of the possible BLE architected advertisement fields are currently exposed. Others will - // be exposed on demand/request or as time permits. - // public: + /*************************************************************************** + * Common public declarations * + ***************************************************************************/ + void setAppearance(uint16_t appearance); void setCompleteServices(BLEUUID uuid); void setFlags(uint8_t); @@ -33,85 +114,144 @@ class BLEAdvertisementData { void setPartialServices(BLEUUID uuid); void setServiceData(BLEUUID uuid, String data); void setShortName(String name); - void addData(String data); // Add data to the payload. - String getPayload(); // Retrieve the current advert payload. + void setPreferredParams(uint16_t min, uint16_t max); + void addTxPower(); + void addData(String data); + void addData(char *data, size_t length); + String getPayload(); private: friend class BLEAdvertising; - String m_payload; // The payload of the advertisement. -}; // BLEAdvertisementData + + /*************************************************************************** + * Common private declarations * + ***************************************************************************/ + + String m_payload; +}; /** * @brief Perform and manage %BLE advertising. - * - * A %BLE server will want to perform advertising in order to make itself known to %BLE clients. */ class BLEAdvertising { public: + /*************************************************************************** + * Common public declarations * + ***************************************************************************/ + BLEAdvertising(); void addServiceUUID(BLEUUID serviceUUID); void addServiceUUID(const char *serviceUUID); bool removeServiceUUID(int index); bool removeServiceUUID(BLEUUID serviceUUID); bool removeServiceUUID(const char *serviceUUID); - bool start(); bool stop(); + void reset(); void setAppearance(uint16_t appearance); - void setAdvertisementType(esp_ble_adv_type_t adv_type); - void setAdvertisementChannelMap(esp_ble_adv_channel_t channel_map); + void setAdvertisementType(uint8_t adv_type); void setMaxInterval(uint16_t maxinterval); void setMinInterval(uint16_t mininterval); bool setAdvertisementData(BLEAdvertisementData &advertisementData); void setScanFilter(bool scanRequestWhitelistOnly, bool connectWhitelistOnly); bool setScanResponseData(BLEAdvertisementData &advertisementData); - void setPrivateAddress(esp_ble_addr_type_t type = BLE_ADDR_TYPE_RANDOM); - bool setDeviceAddress(esp_bd_addr_t addr, esp_ble_addr_type_t type = BLE_ADDR_TYPE_RANDOM); - - void handleGAPEvent(esp_gap_ble_cb_event_t event, esp_ble_gap_cb_param_t *param); void setMinPreferred(uint16_t); void setMaxPreferred(uint16_t); void setScanResponse(bool); + /*************************************************************************** + * Bluedroid public declarations * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) + void setPrivateAddress(esp_ble_addr_type_t type = BLE_ADDR_TYPE_RANDOM); + bool setDeviceAddress(esp_bd_addr_t addr, esp_ble_addr_type_t type = BLE_ADDR_TYPE_RANDOM); + void setAdvertisementChannelMap(esp_ble_adv_channel_t channel_map); + void handleGAPEvent(esp_gap_ble_cb_event_t event, esp_ble_gap_cb_param_t *param); + bool start(); +#endif + + /*************************************************************************** + * NimBLE public declarations * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) + void setName(String name); + void addTxPower(); + void advCompleteCB(); + bool isAdvertising(); + void onHostSync(); + bool start(uint32_t duration = 0, void (*advCompleteCB)(BLEAdvertising *pAdv) = nullptr); + static int handleGAPEvent(ble_gap_event *event, void *arg); +#endif + private: - esp_ble_adv_data_t m_advData; - esp_ble_adv_data_t m_scanRespData; // Used for configuration of scan response data when m_scanResp is true - esp_ble_adv_params_t m_advParams; + /*************************************************************************** + * Common private properties * + ***************************************************************************/ + std::vector m_serviceUUIDs; - bool m_customAdvData = false; // Are we using custom advertising data? - bool m_customScanResponseData = false; // Are we using custom scan response data? + bool m_customAdvData = false; + bool m_customScanResponseData = false; FreeRTOS::Semaphore m_semaphoreSetAdv = FreeRTOS::Semaphore("startAdvert"); bool m_scanResp = true; + + /*************************************************************************** + * Bluedroid private properties * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) + esp_ble_adv_data_t m_advData; + esp_ble_adv_data_t m_scanRespData; + esp_ble_adv_params_t m_advParams; +#endif + + /*************************************************************************** + * NimBLE private properties * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) + ble_hs_adv_fields m_advData; + ble_hs_adv_fields m_scanData; + ble_gap_adv_params m_advParams; + bool m_advDataSet; + void (*m_advCompCB)(BLEAdvertising *pAdv); + uint8_t m_slaveItvl[4]; + uint32_t m_duration; + String m_name; +#endif }; -#ifdef SOC_BLE_50_SUPPORTED +/*************************************************************************** + * Bluedroid 5.0 specific classes * + ***************************************************************************/ +#if defined(SOC_BLE_50_SUPPORTED) && defined(CONFIG_BLUEDROID_ENABLED) class BLEMultiAdvertising { -private: - esp_ble_gap_ext_adv_params_t *params_arrays; - esp_ble_gap_ext_adv_t *ext_adv; - uint8_t count; - public: BLEMultiAdvertising(uint8_t num = 1); ~BLEMultiAdvertising() {} - bool setAdvertisingParams(uint8_t instance, const esp_ble_gap_ext_adv_params_t *params); bool setAdvertisingData(uint8_t instance, uint16_t length, const uint8_t *data); bool setScanRspData(uint8_t instance, uint16_t length, const uint8_t *data); bool start(); bool start(uint8_t num, uint8_t from); void setDuration(uint8_t instance, int duration = 0, int max_events = 0); - bool setInstanceAddress(uint8_t instance, esp_bd_addr_t rand_addr); + bool setInstanceAddress(uint8_t instance, uint8_t *rand_addr); bool stop(uint8_t num_adv, const uint8_t *ext_adv_inst); bool remove(uint8_t instance); bool clear(); - bool setPeriodicAdvertisingParams(uint8_t instance, const esp_ble_gap_periodic_adv_params_t *params); bool setPeriodicAdvertisingData(uint8_t instance, uint16_t length, const uint8_t *data); bool startPeriodicAdvertising(uint8_t instance); -}; + bool setAdvertisingParams(uint8_t instance, const esp_ble_gap_ext_adv_params_t *params); + bool setPeriodicAdvertisingParams(uint8_t instance, const esp_ble_gap_periodic_adv_params_t *params); -#endif // SOC_BLE_50_SUPPORTED +private: + esp_ble_gap_ext_adv_params_t *params_arrays; + esp_ble_gap_ext_adv_t *ext_adv; + uint8_t count; +}; +#endif -#endif /* CONFIG_BLUEDROID_ENABLED */ +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ #endif /* COMPONENTS_CPP_UTILS_BLEADVERTISING_H_ */ diff --git a/libraries/BLE/src/BLEBeacon.cpp b/libraries/BLE/src/BLEBeacon.cpp index 43366a7b2d9..0a6c6b05258 100644 --- a/libraries/BLE/src/BLEBeacon.cpp +++ b/libraries/BLE/src/BLEBeacon.cpp @@ -3,17 +3,31 @@ * * Created on: Jan 4, 2018 * Author: kolban + * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on kolban's and h2zero's work) + * Description: Added support for NimBLE */ + #include "soc/soc_caps.h" #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) + +/*************************************************************************** + * Common includes and definitions * + ***************************************************************************/ + #include "BLEBeacon.h" #include "esp32-hal-log.h" #define ENDIAN_CHANGE_U16(x) ((((x) & 0xFF00) >> 8) + (((x) & 0xFF) << 8)) +/*************************************************************************** + * Common functions * + ***************************************************************************/ + BLEBeacon::BLEBeacon() { m_beaconData.manufacturerId = 0x4c00; m_beaconData.subType = 0x02; @@ -22,11 +36,11 @@ BLEBeacon::BLEBeacon() { m_beaconData.minor = 0; m_beaconData.signalPower = 0; memset(m_beaconData.proximityUUID, 0, sizeof(m_beaconData.proximityUUID)); -} // BLEBeacon +} String BLEBeacon::getData() { return String((char *)&m_beaconData, sizeof(m_beaconData)); -} // getData +} uint16_t BLEBeacon::getMajor() { return m_beaconData.major; @@ -48,37 +62,38 @@ int8_t BLEBeacon::getSignalPower() { return m_beaconData.signalPower; } -/** - * Set the raw data for the beacon record. - */ void BLEBeacon::setData(String data) { if (data.length() != sizeof(m_beaconData)) { log_e("Unable to set the data ... length passed in was %d and expected %d", data.length(), sizeof(m_beaconData)); return; } memcpy(&m_beaconData, data.c_str(), sizeof(m_beaconData)); -} // setData +} void BLEBeacon::setMajor(uint16_t major) { m_beaconData.major = ENDIAN_CHANGE_U16(major); -} // setMajor +} void BLEBeacon::setManufacturerId(uint16_t manufacturerId) { m_beaconData.manufacturerId = ENDIAN_CHANGE_U16(manufacturerId); -} // setManufacturerId +} void BLEBeacon::setMinor(uint16_t minor) { m_beaconData.minor = ENDIAN_CHANGE_U16(minor); -} // setMinior - -void BLEBeacon::setProximityUUID(BLEUUID uuid) { - uuid = uuid.to128(); - memcpy(m_beaconData.proximityUUID, uuid.getNative()->uuid.uuid128, 16); -} // setProximityUUID +} void BLEBeacon::setSignalPower(int8_t signalPower) { m_beaconData.signalPower = signalPower; -} // setSignalPower +} +void BLEBeacon::setProximityUUID(BLEUUID uuid) { + uuid = uuid.to128(); +#if defined(CONFIG_BLUEDROID_ENABLED) + memcpy(m_beaconData.proximityUUID, uuid.getNative()->uuid.uuid128, 16); +#elif defined(CONFIG_NIMBLE_ENABLED) + memcpy(m_beaconData.proximityUUID, uuid.getNative()->u128.value, 16); #endif +} + +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ diff --git a/libraries/BLE/src/BLEBeacon.h b/libraries/BLE/src/BLEBeacon.h index dcc41aafeb4..e0eaddda6e4 100644 --- a/libraries/BLE/src/BLEBeacon.h +++ b/libraries/BLE/src/BLEBeacon.h @@ -3,6 +3,10 @@ * * Created on: Jan 4, 2018 * Author: kolban + * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on kolban's and h2zero's work) + * Description: Added support for NimBLE */ #ifndef COMPONENTS_CPP_UTILS_BLEBEACON_H_ @@ -10,6 +14,13 @@ #include "soc/soc_caps.h" #if SOC_BLE_SUPPORTED +#include "sdkconfig.h" +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) + +/*************************************************************************** + * Common includes * + ***************************************************************************/ + #include "BLEUUID.h" /** * @brief Representation of a beacon. @@ -18,6 +29,10 @@ */ class BLEBeacon { private: + /*************************************************************************** + * Common types * + ***************************************************************************/ + struct { uint16_t manufacturerId; uint8_t subType; @@ -29,6 +44,10 @@ class BLEBeacon { } __attribute__((packed)) m_beaconData; public: + /*************************************************************************** + * Common public declarations * + ***************************************************************************/ + BLEBeacon(); String getData(); uint16_t getMajor(); @@ -44,5 +63,6 @@ class BLEBeacon { void setSignalPower(int8_t signalPower); }; // BLEBeacon +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ #endif /* COMPONENTS_CPP_UTILS_BLEBEACON_H_ */ diff --git a/libraries/BLE/src/BLECharacteristic.cpp b/libraries/BLE/src/BLECharacteristic.cpp index b03d524a6a5..0234cd11cca 100644 --- a/libraries/BLE/src/BLECharacteristic.cpp +++ b/libraries/BLE/src/BLECharacteristic.cpp @@ -3,12 +3,22 @@ * * Created on: Jun 22, 2017 * Author: kolban + * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on kolban's and h2zero's work) + * Description: Added support for NimBLE */ + #include "soc/soc_caps.h" #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) + +/*************************************************************************** + * Common includes * + ***************************************************************************/ + #include #include #include @@ -23,10 +33,31 @@ #include "GeneralUtils.h" #include "esp32-hal-log.h" +/*************************************************************************** + * Common definitions * + ***************************************************************************/ + #define NULL_HANDLE (0xffff) +/*************************************************************************** + * NimBLE definitions * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) +#define NIMBLE_SUB_NOTIFY 0x0001 +#define NIMBLE_SUB_INDICATE 0x0002 +#endif + +/*************************************************************************** + * Common global variables * + ***************************************************************************/ + static BLECharacteristicCallbacks defaultCallback; //null-object-pattern +/*************************************************************************** + * Common functions * + ***************************************************************************/ + /** * @brief Construct a characteristic * @param [in] uuid - UUID (const char*) for the characteristic. @@ -42,15 +73,23 @@ BLECharacteristic::BLECharacteristic(const char *uuid, uint32_t properties) : BL BLECharacteristic::BLECharacteristic(BLEUUID uuid, uint32_t properties) { m_bleUUID = uuid; m_handle = NULL_HANDLE; - m_properties = (esp_gatt_char_prop_t)0; m_pCallbacks = &defaultCallback; - setBroadcastProperty((properties & PROPERTY_BROADCAST) != 0); - setReadProperty((properties & PROPERTY_READ) != 0); - setWriteProperty((properties & PROPERTY_WRITE) != 0); - setNotifyProperty((properties & PROPERTY_NOTIFY) != 0); - setIndicateProperty((properties & PROPERTY_INDICATE) != 0); - setWriteNoResponseProperty((properties & PROPERTY_WRITE_NR) != 0); +#ifdef CONFIG_BLUEDROID_ENABLED + m_properties = 0; + setBroadcastProperty((properties & BLECharacteristic::PROPERTY_BROADCAST) != 0); + setReadProperty((properties & BLECharacteristic::PROPERTY_READ) != 0); + setWriteProperty((properties & BLECharacteristic::PROPERTY_WRITE) != 0); + setNotifyProperty((properties & BLECharacteristic::PROPERTY_NOTIFY) != 0); + setIndicateProperty((properties & BLECharacteristic::PROPERTY_INDICATE) != 0); + setWriteNoResponseProperty((properties & BLECharacteristic::PROPERTY_WRITE_NR) != 0); +#endif + +#if defined(CONFIG_NIMBLE_ENABLED) + m_readMux = portMUX_INITIALIZER_UNLOCKED; + m_removed = 0; + m_properties = properties; +#endif } // BLECharacteristic /** @@ -66,51 +105,20 @@ BLECharacteristic::~BLECharacteristic() { * @return N/A. */ void BLECharacteristic::addDescriptor(BLEDescriptor *pDescriptor) { +#ifdef CONFIG_NIMBLE_ENABLED + if (pDescriptor->getUUID() == BLEUUID(uint16_t(0x2902))) { + log_i("NimBLE automatically creates the 0x2902 descriptor if a characteristic has a notification or indication property assigned to it.\n" + "You should check the characteristic properties for notification or indication rather than adding the descriptor manually.\n" + "This will be removed in a future version of the library."); + pDescriptor->executeCreate(this); + return; + } +#endif log_v(">> addDescriptor(): Adding %s to %s", pDescriptor->toString().c_str(), toString().c_str()); m_descriptorMap.setByUUID(pDescriptor->getUUID(), pDescriptor); log_v("<< addDescriptor()"); } // addDescriptor -/** - * @brief Register a new characteristic with the ESP runtime. - * @param [in] pService The service with which to associate this characteristic. - */ -void BLECharacteristic::executeCreate(BLEService *pService) { - log_v(">> executeCreate()"); - - if (m_handle != NULL_HANDLE) { - log_e("Characteristic already has a handle."); - return; - } - - m_pService = pService; // Save the service to which this characteristic belongs. - - log_d("Registering characteristic (esp_ble_gatts_add_char): uuid: %s, service: %s", getUUID().toString().c_str(), m_pService->toString().c_str()); - - esp_attr_control_t control; - control.auto_rsp = ESP_GATT_RSP_BY_APP; - - m_semaphoreCreateEvt.take("executeCreate"); - esp_err_t errRc = ::esp_ble_gatts_add_char( - m_pService->getHandle(), getUUID().getNative(), static_cast(m_permissions), getProperties(), nullptr, - &control - ); // Whether to auto respond or not. - - if (errRc != ESP_OK) { - log_e("<< esp_ble_gatts_add_char: rc=%d %s", errRc, GeneralUtils::errorToString(errRc)); - return; - } - m_semaphoreCreateEvt.wait("executeCreate"); - - BLEDescriptor *pDescriptor = m_descriptorMap.getFirst(); - while (pDescriptor != nullptr) { - pDescriptor->executeCreate(this); - pDescriptor = m_descriptorMap.getNext(); - } // End while - - log_v("<< executeCreate"); -} // executeCreate - /** * @brief Return the BLE Descriptor for the given UUID if associated with this characteristic. * @param [in] descriptorUUID The UUID of the descriptor that we wish to retrieve. @@ -137,8 +145,10 @@ uint16_t BLECharacteristic::getHandle() { return m_handle; } // getHandle -void BLECharacteristic::setAccessPermissions(esp_gatt_perm_t perm) { +void BLECharacteristic::setAccessPermissions(uint8_t perm) { +#ifdef CONFIG_BLUEDROID_ENABLED m_permissions = perm; +#endif } esp_gatt_char_prop_t BLECharacteristic::getProperties() { @@ -185,390 +195,76 @@ size_t BLECharacteristic::getLength() { } // getLength /** - * Handle a GATT server event. + * @brief Register a new characteristic with the ESP runtime. + * @param [in] pService The service with which to associate this characteristic. */ -void BLECharacteristic::handleGATTServerEvent(esp_gatts_cb_event_t event, esp_gatt_if_t gatts_if, esp_ble_gatts_cb_param_t *param) { - log_v(">> handleGATTServerEvent: %s", BLEUtils::gattServerEventTypeToString(event).c_str()); +void BLECharacteristic::executeCreate(BLEService *pService) { + log_v(">> executeCreate()"); - switch (event) { - // Events handled: - // - // ESP_GATTS_ADD_CHAR_EVT - // ESP_GATTS_CONF_EVT - // ESP_GATTS_CONNECT_EVT - // ESP_GATTS_DISCONNECT_EVT - // ESP_GATTS_EXEC_WRITE_EVT - // ESP_GATTS_READ_EVT - // ESP_GATTS_WRITE_EVT + if (m_handle != NULL_HANDLE) { + log_e("Characteristic already has a handle."); + return; + } - // - // ESP_GATTS_EXEC_WRITE_EVT - // When we receive this event it is an indication that a previous write long needs to be committed. - // - // exec_write: - // - uint16_t conn_id - // - uint32_t trans_id - // - esp_bd_addr_t bda - // - uint8_t exec_write_flag - Either ESP_GATT_PREP_WRITE_EXEC or ESP_GATT_PREP_WRITE_CANCEL - // - case ESP_GATTS_EXEC_WRITE_EVT: - { - if (m_writeEvt) { - m_writeEvt = false; - if (param->exec_write.exec_write_flag == ESP_GATT_PREP_WRITE_EXEC) { - m_value.commit(); - // Invoke the onWrite callback handler. - m_pCallbacks->onWrite(this, param); - } else { - m_value.cancel(); - } - // ??? - esp_err_t errRc = ::esp_ble_gatts_send_response(gatts_if, param->write.conn_id, param->write.trans_id, ESP_GATT_OK, nullptr); - if (errRc != ESP_OK) { - log_e("esp_ble_gatts_send_response: rc=%d %s", errRc, GeneralUtils::errorToString(errRc)); - } - } - break; - } // ESP_GATTS_EXEC_WRITE_EVT + m_pService = pService; // Save the service to which this characteristic belongs. - // ESP_GATTS_ADD_CHAR_EVT - Indicate that a characteristic was added to the service. - // add_char: - // - esp_gatt_status_t status - // - uint16_t attr_handle - // - uint16_t service_handle - // - esp_bt_uuid_t char_uuid - case ESP_GATTS_ADD_CHAR_EVT: - { - if (getHandle() == param->add_char.attr_handle) { - // we have created characteristic, now we can create descriptors - // BLEDescriptor* pDescriptor = m_descriptorMap.getFirst(); - // while (pDescriptor != nullptr) { - // pDescriptor->executeCreate(this); - // pDescriptor = m_descriptorMap.getNext(); - // } // End while - m_semaphoreCreateEvt.give(); - } - break; - } // ESP_GATTS_ADD_CHAR_EVT +#ifdef CONFIG_BLUEDROID_ENABLED + log_d("Registering characteristic (esp_ble_gatts_add_char): uuid: %s, service: %s", getUUID().toString().c_str(), m_pService->toString().c_str()); - // ESP_GATTS_WRITE_EVT - A request to write the value of a characteristic has arrived. - // - // write: - // - uint16_t conn_id - // - uint16_t trans_id - // - esp_bd_addr_t bda - // - uint16_t handle - // - uint16_t offset - // - bool need_rsp - // - bool is_prep - // - uint16_t len - // - uint8_t *value - // - case ESP_GATTS_WRITE_EVT: - { - // We check if this write request is for us by comparing the handles in the event. If it is for us - // we save the new value. Next we look at the need_rsp flag which indicates whether or not we need - // to send a response. If we do, then we formulate a response and send it. - if (param->write.handle == m_handle) { - if (param->write.is_prep) { - m_value.addPart(param->write.value, param->write.len); - m_writeEvt = true; - } else { - setValue(param->write.value, param->write.len); - } + esp_attr_control_t control; + control.auto_rsp = ESP_GATT_RSP_BY_APP; - log_d(" - Response to write event: New value: handle: %.2x, uuid: %s", getHandle(), getUUID().toString().c_str()); + m_semaphoreCreateEvt.take("executeCreate"); + esp_err_t errRc = ::esp_ble_gatts_add_char( + m_pService->getHandle(), getUUID().getNative(), static_cast(m_permissions), getProperties(), nullptr, + &control + ); // Whether to auto respond or not. + + if (errRc != ESP_OK) { + log_e("<< esp_ble_gatts_add_char: rc=%d %s", errRc, GeneralUtils::errorToString(errRc)); + return; + } + m_semaphoreCreateEvt.wait("executeCreate"); -// The call to BLEUtils::buildHexData() doesn't output anything if the log level is not -// "DEBUG". As it is quite CPU intensive, it is much better to not call it if not needed. -#if ARDUHAL_LOG_LEVEL >= ARDUHAL_LOG_LEVEL_DEBUG - char *pHexData = BLEUtils::buildHexData(nullptr, param->write.value, param->write.len); - log_d(" - Data: length: %d, data: %s", param->write.len, pHexData); - free(pHexData); #endif - if (param->write.need_rsp) { - esp_gatt_rsp_t rsp; + BLEDescriptor *pDescriptor = m_descriptorMap.getFirst(); + while (pDescriptor != nullptr) { + pDescriptor->executeCreate(this); + pDescriptor = m_descriptorMap.getNext(); + } // End while - rsp.attr_value.len = param->write.len; - rsp.attr_value.handle = m_handle; - rsp.attr_value.offset = param->write.offset; - rsp.attr_value.auth_req = ESP_GATT_AUTH_REQ_NONE; - memcpy(rsp.attr_value.value, param->write.value, param->write.len); + log_v("<< executeCreate"); +} // executeCreate - esp_err_t errRc = ::esp_ble_gatts_send_response(gatts_if, param->write.conn_id, param->write.trans_id, ESP_GATT_OK, &rsp); - if (errRc != ESP_OK) { - log_e("esp_ble_gatts_send_response: rc=%d %s", errRc, GeneralUtils::errorToString(errRc)); - } - } // Response needed +/** + * @brief Send an indication. + * An indication is a transmission of up to the first 20 bytes of the characteristic value. An indication + * will block waiting a positive confirmation from the client. + * @return N/A + */ +void BLECharacteristic::indicate() { - if (param->write.is_prep != true) { - // Invoke the onWrite callback handler. - m_pCallbacks->onWrite(this, param); - } - } // Match on handles. - break; - } // ESP_GATTS_WRITE_EVT + log_v(">> indicate: length: %d", m_value.getValue().length()); + notify(false); + log_v("<< indicate"); +} // indicate - // ESP_GATTS_READ_EVT - A request to read the value of a characteristic has arrived. - // - // read: - // - uint16_t conn_id - // - uint32_t trans_id - // - esp_bd_addr_t bda - // - uint16_t handle - // - uint16_t offset - // - bool is_long - // - bool need_rsp - // - case ESP_GATTS_READ_EVT: - { - if (param->read.handle == m_handle) { - - // Here's an interesting thing. The read request has the option of saying whether we need a response - // or not. What would it "mean" to receive a read request and NOT send a response back? That feels like - // a very strange read. - // - // We have to handle the case where the data we wish to send back to the client is greater than the maximum - // packet size of 22 bytes. In this case, we become responsible for chunking the data into units of 22 bytes. - // The apparent algorithm is as follows: - // - // If the is_long flag is set then this is a follow on from an original read and we will already have sent at least 22 bytes. - // If the is_long flag is not set then we need to check how much data we are going to send. If we are sending LESS than - // 22 bytes, then we "just" send it and that's the end of the story. - // If we are sending 22 bytes exactly, we just send it BUT we will get a follow on request. - // If we are sending more than 22 bytes, we send the first 22 bytes and we will get a follow on request. - // Because of follow on request processing, we need to maintain an offset of how much data we have already sent - // so that when a follow on request arrives, we know where to start in the data to send the next sequence. - // Note that the indication that the client will send a follow on request is that we sent exactly 22 bytes as a response. - // If our payload is divisible by 22 then the last response will be a response of 0 bytes in length. - // - // The following code has deliberately not been factored to make it fewer statements because this would cloud the - // the logic flow comprehension. - // - - // get mtu for peer device that we are sending read request to - uint16_t maxOffset = getService()->getServer()->getPeerMTU(param->read.conn_id) - 1; - log_d("mtu value: %d", maxOffset); - if (param->read.need_rsp) { - log_d("Sending a response (esp_ble_gatts_send_response)"); - esp_gatt_rsp_t rsp; - - if (param->read.is_long) { - String value = m_value.getValue(); - - if (value.length() - m_value.getReadOffset() < maxOffset) { - // This is the last in the chain - rsp.attr_value.len = value.length() - m_value.getReadOffset(); - rsp.attr_value.offset = m_value.getReadOffset(); - memcpy(rsp.attr_value.value, value.c_str() + rsp.attr_value.offset, rsp.attr_value.len); - m_value.setReadOffset(0); - } else { - // There will be more to come. - rsp.attr_value.len = maxOffset; - rsp.attr_value.offset = m_value.getReadOffset(); - memcpy(rsp.attr_value.value, value.c_str() + rsp.attr_value.offset, rsp.attr_value.len); - m_value.setReadOffset(rsp.attr_value.offset + maxOffset); - } - } else { // read.is_long == false - - // If is.long is false then this is the first (or only) request to read data, so invoke the callback - // Invoke the read callback. - m_pCallbacks->onRead(this, param); - - String value = m_value.getValue(); - - if (value.length() + 1 > maxOffset) { - // Too big for a single shot entry. - m_value.setReadOffset(maxOffset); - rsp.attr_value.len = maxOffset; - rsp.attr_value.offset = 0; - memcpy(rsp.attr_value.value, value.c_str(), rsp.attr_value.len); - } else { - // Will fit in a single packet with no callbacks required. - rsp.attr_value.len = value.length(); - rsp.attr_value.offset = 0; - memcpy(rsp.attr_value.value, value.c_str(), rsp.attr_value.len); - } - } - rsp.attr_value.handle = param->read.handle; - rsp.attr_value.auth_req = ESP_GATT_AUTH_REQ_NONE; - -// The call to BLEUtils::buildHexData() doesn't output anything if the log level is not -// "DEBUG". As it is quite CPU intensive, it is much better to not call it if not needed. -#if ARDUHAL_LOG_LEVEL >= ARDUHAL_LOG_LEVEL_DEBUG - char *pHexData = BLEUtils::buildHexData(nullptr, rsp.attr_value.value, rsp.attr_value.len); - log_d(" - Data: length=%d, data=%s, offset=%d", rsp.attr_value.len, pHexData, rsp.attr_value.offset); - free(pHexData); -#endif - - esp_err_t errRc = ::esp_ble_gatts_send_response(gatts_if, param->read.conn_id, param->read.trans_id, ESP_GATT_OK, &rsp); - if (errRc != ESP_OK) { - log_e("esp_ble_gatts_send_response: rc=%d %s", errRc, GeneralUtils::errorToString(errRc)); - } - } // Response needed - } // Handle matches this characteristic. - break; - } // ESP_GATTS_READ_EVT - - // ESP_GATTS_CONF_EVT - // - // conf: - // - esp_gatt_status_t status – The status code. - // - uint16_t conn_id – The connection used. - // - case ESP_GATTS_CONF_EVT: - { - // log_d("m_handle = %d, conf->handle = %d", m_handle, param->conf.handle); - if (param->conf.conn_id - == getService()->getServer()->getConnId()) { // && param->conf.handle == m_handle) // bug in esp-idf and not implemented in arduino yet - m_semaphoreConfEvt.give(param->conf.status); - } - break; - } - - case ESP_GATTS_CONNECT_EVT: - { - break; - } - - case ESP_GATTS_DISCONNECT_EVT: - { - m_semaphoreConfEvt.give(); - break; - } - - default: - { - break; - } // default - - } // switch event - - // Give each of the descriptors associated with this characteristic the opportunity to handle the - // event. - - m_descriptorMap.handleGATTServerEvent(event, gatts_if, param); - log_v("<< handleGATTServerEvent"); -} // handleGATTServerEvent - -/** - * @brief Send an indication. - * An indication is a transmission of up to the first 20 bytes of the characteristic value. An indication - * will block waiting a positive confirmation from the client. - * @return N/A - */ -void BLECharacteristic::indicate() { - - log_v(">> indicate: length: %d", m_value.getValue().length()); - notify(false); - log_v("<< indicate"); -} // indicate - -/** - * @brief Send a notify. - * A notification is a transmission of up to the first 20 bytes of the characteristic value. An notification - * will not block; it is a fire and forget. - * @return N/A. - */ -void BLECharacteristic::notify(bool is_notification) { - log_v(">> notify: length: %d", m_value.getValue().length()); - - assert(getService() != nullptr); - assert(getService()->getServer() != nullptr); - - m_pCallbacks->onNotify(this); // Invoke the notify callback. - - // GeneralUtils::hexDump() doesn't output anything if the log level is not - // "VERBOSE". Additionally, it is very CPU intensive, even when it doesn't - // output anything! So it is much better to *not* call it at all if not needed. - // In a simple program which calls BLECharacteristic::notify() every 50 ms, - // the performance gain of this little optimization is 37% in release mode - // (-O3) and 57% in debug mode. - // Of course, the "#if ARDUHAL_LOG_LEVEL >= ARDUHAL_LOG_LEVEL_VERBOSE" guard - // could also be put inside the GeneralUtils::hexDump() function itself. But - // it's better to put it here also, as it is clearer (indicating a verbose log - // thing) and it allows to remove the "m_value.getValue().c_str()" call, which - // is, in itself, quite CPU intensive. -#if ARDUHAL_LOG_LEVEL >= ARDUHAL_LOG_LEVEL_VERBOSE - GeneralUtils::hexDump((uint8_t *)m_value.getValue().c_str(), m_value.getValue().length()); -#endif - - if (getService()->getServer()->getConnectedCount() == 0) { - log_v("<< notify: No connected clients."); - m_pCallbacks->onStatus(this, BLECharacteristicCallbacks::Status::ERROR_NO_CLIENT, 0); - return; - } - - // Test to see if we have a 0x2902 descriptor. If we do, then check to see if notification is enabled - // and, if not, prevent the notification. - - BLE2902 *p2902 = (BLE2902 *)getDescriptorByUUID((uint16_t)0x2902); - if (is_notification) { - if (p2902 != nullptr && !p2902->getNotifications()) { - log_v("<< notifications disabled; ignoring"); - m_pCallbacks->onStatus(this, BLECharacteristicCallbacks::Status::ERROR_NOTIFY_DISABLED, 0); // Invoke the notify callback. - return; - } - } else { - if (p2902 != nullptr && !p2902->getIndications()) { - log_v("<< indications disabled; ignoring"); - m_pCallbacks->onStatus(this, BLECharacteristicCallbacks::Status::ERROR_INDICATE_DISABLED, 0); // Invoke the notify callback. - return; - } - } - for (auto &myPair : getService()->getServer()->getPeerDevices(false)) { - uint16_t _mtu = (myPair.second.mtu); - if (m_value.getValue().length() > _mtu - 3) { - log_w("- Truncating to %d bytes (maximum notify size)", _mtu - 3); - } - - size_t length = m_value.getValue().length(); - if (!is_notification) { // is indication - m_semaphoreConfEvt.take("indicate"); - } - esp_err_t errRc = ::esp_ble_gatts_send_indicate( - getService()->getServer()->getGattsIf(), myPair.first, getHandle(), length, (uint8_t *)m_value.getValue().c_str(), !is_notification - ); // The need_confirm = false makes this a notify. - if (errRc != ESP_OK) { - log_e("<< esp_ble_gatts_send_ %s: rc=%d %s", is_notification ? "notify" : "indicate", errRc, GeneralUtils::errorToString(errRc)); - m_semaphoreConfEvt.give(); - m_pCallbacks->onStatus(this, BLECharacteristicCallbacks::Status::ERROR_GATT, errRc); // Invoke the notify callback. - return; - } - if (!is_notification) { // is indication - if (!m_semaphoreConfEvt.timedWait("indicate", indicationTimeout)) { - m_pCallbacks->onStatus(this, BLECharacteristicCallbacks::Status::ERROR_INDICATE_TIMEOUT, 0); // Invoke the notify callback. - } else { - auto code = (esp_gatt_status_t)m_semaphoreConfEvt.value(); - if (code == ESP_GATT_OK) { - m_pCallbacks->onStatus(this, BLECharacteristicCallbacks::Status::SUCCESS_INDICATE, code); // Invoke the notify callback. - } else { - m_pCallbacks->onStatus(this, BLECharacteristicCallbacks::Status::ERROR_INDICATE_FAILURE, code); - } - } - } else { - m_pCallbacks->onStatus(this, BLECharacteristicCallbacks::Status::SUCCESS_NOTIFY, 0); // Invoke the notify callback. - } - } - log_v("<< notify"); -} // Notify - -/** - * @brief Set the permission to broadcast. - * A characteristics has properties associated with it which define what it is capable of doing. - * One of these is the broadcast flag. - * @param [in] value The flag value of the property. - * @return N/A - */ -void BLECharacteristic::setBroadcastProperty(bool value) { - //log_d("setBroadcastProperty(%d)", value); - if (value) { - m_properties = (esp_gatt_char_prop_t)(m_properties | ESP_GATT_CHAR_PROP_BIT_BROADCAST); - } else { - m_properties = (esp_gatt_char_prop_t)(m_properties & ~ESP_GATT_CHAR_PROP_BIT_BROADCAST); - } -} // setBroadcastProperty +/** + * @brief Set the permission to broadcast. + * A characteristics has properties associated with it which define what it is capable of doing. + * One of these is the broadcast flag. + * @param [in] value The flag value of the property. + * @return N/A + */ +void BLECharacteristic::setBroadcastProperty(bool value) { + //log_d("setBroadcastProperty(%d)", value); + if (value) { + m_properties = (esp_gatt_char_prop_t)(m_properties | ESP_GATT_CHAR_PROP_BIT_BROADCAST); + } else { + m_properties = (esp_gatt_char_prop_t)(m_properties & ~ESP_GATT_CHAR_PROP_BIT_BROADCAST); + } +} // setBroadcastProperty /** * @brief Set the callback handlers for this characteristic. @@ -595,9 +291,15 @@ void BLECharacteristic::setCallbacks(BLECharacteristicCallbacks *pCallbacks) { * @param [in] handle The handle associated with this characteristic. */ void BLECharacteristic::setHandle(uint16_t handle) { +#if defined(CONFIG_BLUEDROID_ENABLED) log_v(">> setHandle: handle=0x%.2x, characteristic uuid=%s", handle, getUUID().toString().c_str()); m_handle = handle; log_v("<< setHandle"); +#endif + +#if defined(CONFIG_NIMBLE_ENABLED) + log_w("NimBLE does not support manually setting the handle of a characteristic. Ignoring request."); +#endif } // setHandle /** @@ -767,19 +469,12 @@ String BLECharacteristic::toString() { BLECharacteristicCallbacks::~BLECharacteristicCallbacks() {} -void BLECharacteristicCallbacks::onRead(BLECharacteristic *pCharacteristic, esp_ble_gatts_cb_param_t *param) { - onRead(pCharacteristic); -} // onRead - +// Common callbacks void BLECharacteristicCallbacks::onRead(BLECharacteristic *pCharacteristic) { log_d(">> onRead: default"); log_d("<< onRead"); } // onRead -void BLECharacteristicCallbacks::onWrite(BLECharacteristic *pCharacteristic, esp_ble_gatts_cb_param_t *param) { - onWrite(pCharacteristic); -} // onWrite - void BLECharacteristicCallbacks::onWrite(BLECharacteristic *pCharacteristic) { log_d(">> onWrite: default"); log_d("<< onWrite"); @@ -795,5 +490,650 @@ void BLECharacteristicCallbacks::onStatus(BLECharacteristic *pCharacteristic, St log_d("<< onStatus"); } // onStatus -#endif /* CONFIG_BLUEDROID_ENABLED */ +/*************************************************************************** + * Bluedroid functions * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) + +/** + * Handle a GATT server event. + */ +void BLECharacteristic::handleGATTServerEvent(esp_gatts_cb_event_t event, esp_gatt_if_t gatts_if, esp_ble_gatts_cb_param_t *param) { + log_v(">> handleGATTServerEvent: %s", BLEUtils::gattServerEventTypeToString(event).c_str()); + + switch (event) { + // Events handled: + // + // ESP_GATTS_ADD_CHAR_EVT + // ESP_GATTS_CONF_EVT + // ESP_GATTS_CONNECT_EVT + // ESP_GATTS_DISCONNECT_EVT + // ESP_GATTS_EXEC_WRITE_EVT + // ESP_GATTS_READ_EVT + // ESP_GATTS_WRITE_EVT + + // + // ESP_GATTS_EXEC_WRITE_EVT + // When we receive this event it is an indication that a previous write long needs to be committed. + // + // exec_write: + // - uint16_t conn_id + // - uint32_t trans_id + // - esp_bd_addr_t bda + // - uint8_t exec_write_flag - Either ESP_GATT_PREP_WRITE_EXEC or ESP_GATT_PREP_WRITE_CANCEL + // + case ESP_GATTS_EXEC_WRITE_EVT: + { + if (m_writeEvt) { + m_writeEvt = false; + if (param->exec_write.exec_write_flag == ESP_GATT_PREP_WRITE_EXEC) { + m_value.commit(); + // Invoke the onWrite callback handler. + m_pCallbacks->onWrite(this, param); + } else { + m_value.cancel(); + } + // ??? + esp_err_t errRc = ::esp_ble_gatts_send_response(gatts_if, param->write.conn_id, param->write.trans_id, ESP_GATT_OK, nullptr); + if (errRc != ESP_OK) { + log_e("esp_ble_gatts_send_response: rc=%d %s", errRc, GeneralUtils::errorToString(errRc)); + } + } + break; + } // ESP_GATTS_EXEC_WRITE_EVT + + // ESP_GATTS_ADD_CHAR_EVT - Indicate that a characteristic was added to the service. + // add_char: + // - esp_gatt_status_t status + // - uint16_t attr_handle + // - uint16_t service_handle + // - esp_bt_uuid_t char_uuid + case ESP_GATTS_ADD_CHAR_EVT: + { + if (getHandle() == param->add_char.attr_handle) { + // we have created characteristic, now we can create descriptors + // BLEDescriptor* pDescriptor = m_descriptorMap.getFirst(); + // while (pDescriptor != nullptr) { + // pDescriptor->executeCreate(this); + // pDescriptor = m_descriptorMap.getNext(); + // } // End while + m_semaphoreCreateEvt.give(); + } + break; + } // ESP_GATTS_ADD_CHAR_EVT + + // ESP_GATTS_WRITE_EVT - A request to write the value of a characteristic has arrived. + // + // write: + // - uint16_t conn_id + // - uint16_t trans_id + // - esp_bd_addr_t bda + // - uint16_t handle + // - uint16_t offset + // - bool need_rsp + // - bool is_prep + // - uint16_t len + // - uint8_t *value + // + case ESP_GATTS_WRITE_EVT: + { + // We check if this write request is for us by comparing the handles in the event. If it is for us + // we save the new value. Next we look at the need_rsp flag which indicates whether or not we need + // to send a response. If we do, then we formulate a response and send it. + if (param->write.handle == m_handle) { + if (param->write.is_prep) { + m_value.addPart(param->write.value, param->write.len); + m_writeEvt = true; + } else { + setValue(param->write.value, param->write.len); + } + + log_d(" - Response to write event: New value: handle: %.2x, uuid: %s", getHandle(), getUUID().toString().c_str()); + +// The call to BLEUtils::buildHexData() doesn't output anything if the log level is not +// "DEBUG". As it is quite CPU intensive, it is much better to not call it if not needed. +#if ARDUHAL_LOG_LEVEL >= ARDUHAL_LOG_LEVEL_DEBUG + char *pHexData = BLEUtils::buildHexData(nullptr, param->write.value, param->write.len); + log_d(" - Data: length: %d, data: %s", param->write.len, pHexData); + free(pHexData); +#endif + + if (param->write.need_rsp) { + esp_gatt_rsp_t rsp; + + rsp.attr_value.len = param->write.len; + rsp.attr_value.handle = m_handle; + rsp.attr_value.offset = param->write.offset; + rsp.attr_value.auth_req = ESP_GATT_AUTH_REQ_NONE; + memcpy(rsp.attr_value.value, param->write.value, param->write.len); + + esp_err_t errRc = ::esp_ble_gatts_send_response(gatts_if, param->write.conn_id, param->write.trans_id, ESP_GATT_OK, &rsp); + if (errRc != ESP_OK) { + log_e("esp_ble_gatts_send_response: rc=%d %s", errRc, GeneralUtils::errorToString(errRc)); + } + } // Response needed + + if (param->write.is_prep != true) { + // Invoke the onWrite callback handler. + m_pCallbacks->onWrite(this, param); + } + } // Match on handles. + break; + } // ESP_GATTS_WRITE_EVT + + // ESP_GATTS_READ_EVT - A request to read the value of a characteristic has arrived. + // + // read: + // - uint16_t conn_id + // - uint32_t trans_id + // - esp_bd_addr_t bda + // - uint16_t handle + // - uint16_t offset + // - bool is_long + // - bool need_rsp + // + case ESP_GATTS_READ_EVT: + { + if (param->read.handle == m_handle) { + + // Here's an interesting thing. The read request has the option of saying whether we need a response + // or not. What would it "mean" to receive a read request and NOT send a response back? That feels like + // a very strange read. + // + // We have to handle the case where the data we wish to send back to the client is greater than the maximum + // packet size of 22 bytes. In this case, we become responsible for chunking the data into units of 22 bytes. + // The apparent algorithm is as follows: + // + // If the is_long flag is set then this is a follow on from an original read and we will already have sent at least 22 bytes. + // If the is_long flag is not set then we need to check how much data we are going to send. If we are sending LESS than + // 22 bytes, then we "just" send it and that's the end of the story. + // If we are sending 22 bytes exactly, we just send it BUT we will get a follow on request. + // If we are sending more than 22 bytes, we send the first 22 bytes and we will get a follow on request. + // Because of follow on request processing, we need to maintain an offset of how much data we have already sent + // so that when a follow on request arrives, we know where to start in the data to send the next sequence. + // Note that the indication that the client will send a follow on request is that we sent exactly 22 bytes as a response. + // If our payload is divisible by 22 then the last response will be a response of 0 bytes in length. + // + // The following code has deliberately not been factored to make it fewer statements because this would cloud the + // the logic flow comprehension. + // + + // get mtu for peer device that we are sending read request to + uint16_t maxOffset = getService()->getServer()->getPeerMTU(param->read.conn_id) - 1; + log_d("mtu value: %d", maxOffset); + if (param->read.need_rsp) { + log_d("Sending a response (esp_ble_gatts_send_response)"); + esp_gatt_rsp_t rsp; + + if (param->read.is_long) { + String value = m_value.getValue(); + + if (value.length() - m_value.getReadOffset() < maxOffset) { + // This is the last in the chain + rsp.attr_value.len = value.length() - m_value.getReadOffset(); + rsp.attr_value.offset = m_value.getReadOffset(); + memcpy(rsp.attr_value.value, value.c_str() + rsp.attr_value.offset, rsp.attr_value.len); + m_value.setReadOffset(0); + } else { + // There will be more to come. + rsp.attr_value.len = maxOffset; + rsp.attr_value.offset = m_value.getReadOffset(); + memcpy(rsp.attr_value.value, value.c_str() + rsp.attr_value.offset, rsp.attr_value.len); + m_value.setReadOffset(rsp.attr_value.offset + maxOffset); + } + } else { // read.is_long == false + + // If is.long is false then this is the first (or only) request to read data, so invoke the callback + // Invoke the read callback. + m_pCallbacks->onRead(this, param); + + String value = m_value.getValue(); + + if (value.length() + 1 > maxOffset) { + // Too big for a single shot entry. + m_value.setReadOffset(maxOffset); + rsp.attr_value.len = maxOffset; + rsp.attr_value.offset = 0; + memcpy(rsp.attr_value.value, value.c_str(), rsp.attr_value.len); + } else { + // Will fit in a single packet with no callbacks required. + rsp.attr_value.len = value.length(); + rsp.attr_value.offset = 0; + memcpy(rsp.attr_value.value, value.c_str(), rsp.attr_value.len); + } + } + rsp.attr_value.handle = param->read.handle; + rsp.attr_value.auth_req = ESP_GATT_AUTH_REQ_NONE; + +// The call to BLEUtils::buildHexData() doesn't output anything if the log level is not +// "DEBUG". As it is quite CPU intensive, it is much better to not call it if not needed. +#if ARDUHAL_LOG_LEVEL >= ARDUHAL_LOG_LEVEL_DEBUG + char *pHexData = BLEUtils::buildHexData(nullptr, rsp.attr_value.value, rsp.attr_value.len); + log_d(" - Data: length=%d, data=%s, offset=%d", rsp.attr_value.len, pHexData, rsp.attr_value.offset); + free(pHexData); +#endif + + esp_err_t errRc = ::esp_ble_gatts_send_response(gatts_if, param->read.conn_id, param->read.trans_id, ESP_GATT_OK, &rsp); + if (errRc != ESP_OK) { + log_e("esp_ble_gatts_send_response: rc=%d %s", errRc, GeneralUtils::errorToString(errRc)); + } + } // Response needed + } // Handle matches this characteristic. + break; + } // ESP_GATTS_READ_EVT + + // ESP_GATTS_CONF_EVT + // + // conf: + // - esp_gatt_status_t status – The status code. + // - uint16_t conn_id – The connection used. + // + case ESP_GATTS_CONF_EVT: + { + // log_d("m_handle = %d, conf->handle = %d", m_handle, param->conf.handle); + if (param->conf.conn_id + == getService()->getServer()->getConnId()) { // && param->conf.handle == m_handle) // bug in esp-idf and not implemented in arduino yet + m_semaphoreConfEvt.give(param->conf.status); + } + break; + } + + case ESP_GATTS_CONNECT_EVT: + { + break; + } + + case ESP_GATTS_DISCONNECT_EVT: + { + m_semaphoreConfEvt.give(); + break; + } + + default: + { + break; + } // default + + } // switch event + + // Give each of the descriptors associated with this characteristic the opportunity to handle the + // event. + + m_descriptorMap.handleGATTServerEvent(event, gatts_if, param); + log_v("<< handleGATTServerEvent"); +} // handleGATTServerEvent + +/** + * @brief Send a notify. + * A notification is a transmission of up to the first 20 bytes of the characteristic value. An notification + * will not block; it is a fire and forget. + * @return N/A. + */ +void BLECharacteristic::notify(bool is_notification) { + log_v(">> notify: length: %d", m_value.getValue().length()); + + assert(getService() != nullptr); + assert(getService()->getServer() != nullptr); + + m_pCallbacks->onNotify(this); // Invoke the notify callback. + + // GeneralUtils::hexDump() doesn't output anything if the log level is not + // "VERBOSE". Additionally, it is very CPU intensive, even when it doesn't + // output anything! So it is much better to *not* call it at all if not needed. + // In a simple program which calls BLECharacteristic::notify() every 50 ms, + // the performance gain of this little optimization is 37% in release mode + // (-O3) and 57% in debug mode. + // Of course, the "#if ARDUHAL_LOG_LEVEL >= ARDUHAL_LOG_LEVEL_VERBOSE" guard + // could also be put inside the GeneralUtils::hexDump() function itself. But + // it's better to put it here also, as it is clearer (indicating a verbose log + // thing) and it allows to remove the "m_value.getValue().c_str()" call, which + // is, in itself, quite CPU intensive. +#if ARDUHAL_LOG_LEVEL >= ARDUHAL_LOG_LEVEL_VERBOSE + GeneralUtils::hexDump((uint8_t *)m_value.getValue().c_str(), m_value.getValue().length()); +#endif + + if (getService()->getServer()->getConnectedCount() == 0) { + log_v("<< notify: No connected clients."); + m_pCallbacks->onStatus(this, BLECharacteristicCallbacks::Status::ERROR_NO_CLIENT, 0); + return; + } + + // Test to see if we have a 0x2902 descriptor. If we do, then check to see if notification is enabled + // and, if not, prevent the notification. + + BLE2902 *p2902 = (BLE2902 *)getDescriptorByUUID((uint16_t)0x2902); + if (is_notification) { + if (p2902 != nullptr && !p2902->getNotifications()) { + log_v("<< notifications disabled; ignoring"); + m_pCallbacks->onStatus(this, BLECharacteristicCallbacks::Status::ERROR_NOTIFY_DISABLED, 0); // Invoke the notify callback. + return; + } + } else { + if (p2902 != nullptr && !p2902->getIndications()) { + log_v("<< indications disabled; ignoring"); + m_pCallbacks->onStatus(this, BLECharacteristicCallbacks::Status::ERROR_INDICATE_DISABLED, 0); // Invoke the notify callback. + return; + } + } + for (auto &myPair : getService()->getServer()->getPeerDevices(false)) { + uint16_t _mtu = (myPair.second.mtu); + if (m_value.getValue().length() > _mtu - 3) { + log_w("- Truncating to %d bytes (maximum notify size)", _mtu - 3); + } + + size_t length = m_value.getValue().length(); + if (!is_notification) { // is indication + m_semaphoreConfEvt.take("indicate"); + } + esp_err_t errRc = ::esp_ble_gatts_send_indicate( + getService()->getServer()->getGattsIf(), myPair.first, getHandle(), length, (uint8_t *)m_value.getValue().c_str(), !is_notification + ); // The need_confirm = false makes this a notify. + if (errRc != ESP_OK) { + log_e("<< esp_ble_gatts_send_ %s: rc=%d %s", is_notification ? "notify" : "indicate", errRc, GeneralUtils::errorToString(errRc)); + m_semaphoreConfEvt.give(); + m_pCallbacks->onStatus(this, BLECharacteristicCallbacks::Status::ERROR_GATT, errRc); // Invoke the notify callback. + return; + } + if (!is_notification) { // is indication + if (!m_semaphoreConfEvt.timedWait("indicate", indicationTimeout)) { + m_pCallbacks->onStatus(this, BLECharacteristicCallbacks::Status::ERROR_INDICATE_TIMEOUT, 0); // Invoke the notify callback. + } else { + auto code = (esp_gatt_status_t)m_semaphoreConfEvt.value(); + if (code == ESP_GATT_OK) { + m_pCallbacks->onStatus(this, BLECharacteristicCallbacks::Status::SUCCESS_INDICATE, code); // Invoke the notify callback. + } else { + m_pCallbacks->onStatus(this, BLECharacteristicCallbacks::Status::ERROR_INDICATE_FAILURE, code); + } + } + } else { + m_pCallbacks->onStatus(this, BLECharacteristicCallbacks::Status::SUCCESS_NOTIFY, 0); // Invoke the notify callback. + } + } + log_v("<< notify"); +} // Notify + +void BLECharacteristicCallbacks::onRead(BLECharacteristic *pCharacteristic, esp_ble_gatts_cb_param_t *param) { + onRead(pCharacteristic); +} // onRead + +void BLECharacteristicCallbacks::onWrite(BLECharacteristic *pCharacteristic, esp_ble_gatts_cb_param_t *param) { + onWrite(pCharacteristic); +} // onWrite + +#endif /* CONFIG_BLUEDROID_ENABLED */ + +/*************************************************************************** + * NimBLE functions * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) + +int BLECharacteristic::handleGATTServerEvent(uint16_t conn_handle, uint16_t attr_handle, struct ble_gatt_access_ctxt *ctxt, void *arg) { + const ble_uuid_t *uuid; + int rc; + struct ble_gap_conn_desc desc; + BLECharacteristic *pCharacteristic = (BLECharacteristic *)arg; + + log_d("Characteristic %s %s event", pCharacteristic->getUUID().toString().c_str(), ctxt->op == BLE_GATT_ACCESS_OP_READ_CHR ? "Read" : "Write"); + + uuid = ctxt->chr->uuid; + if (ble_uuid_cmp(uuid, &pCharacteristic->getUUID().getNative()->u) == 0) { + switch (ctxt->op) { + case BLE_GATT_ACCESS_OP_READ_CHR: + { + // If the packet header is only 8 bytes this is a follow up of a long read + // so we don't want to call the onRead() callback again. + if (ctxt->om->om_pkthdr_len > 8) { + rc = ble_gap_conn_find(conn_handle, &desc); + assert(rc == 0); + pCharacteristic->m_pCallbacks->onRead(pCharacteristic); + pCharacteristic->m_pCallbacks->onRead(pCharacteristic, &desc); + } + + portENTER_CRITICAL(&pCharacteristic->m_readMux); + rc = os_mbuf_append(ctxt->om, (uint8_t *)pCharacteristic->m_value.getValue().c_str(), pCharacteristic->m_value.getValue().length()); + portEXIT_CRITICAL(&pCharacteristic->m_readMux); + + return rc == 0 ? 0 : BLE_ATT_ERR_INSUFFICIENT_RES; + } + + case BLE_GATT_ACCESS_OP_WRITE_CHR: + { + if (ctxt->om->om_len > BLE_ATT_ATTR_MAX_LEN) { + return BLE_ATT_ERR_INVALID_ATTR_VALUE_LEN; + } + + uint8_t buf[BLE_ATT_ATTR_MAX_LEN]; + size_t len = ctxt->om->om_len; + memcpy(buf, ctxt->om->om_data, len); + + os_mbuf *next; + next = SLIST_NEXT(ctxt->om, om_next); + while (next != NULL) { + if ((len + next->om_len) > BLE_ATT_ATTR_MAX_LEN) { + return BLE_ATT_ERR_INVALID_ATTR_VALUE_LEN; + } + memcpy(&buf[len], next->om_data, next->om_len); + len += next->om_len; + next = SLIST_NEXT(next, om_next); + } + rc = ble_gap_conn_find(conn_handle, &desc); + assert(rc == 0); + pCharacteristic->setValue(buf, len); + pCharacteristic->m_pCallbacks->onWrite(pCharacteristic); + pCharacteristic->m_pCallbacks->onWrite(pCharacteristic, &desc); + + return 0; + } + + default: break; + } + } + + return BLE_ATT_ERR_UNLIKELY; + + //m_descriptorMap.handleGATTServerEvent(conn_handle, attr_handle, ctxt, arg); +} + +/** + * @brief Set the subscribe status for this characteristic.\n + * This will maintain a vector of subscribed clients and their indicate/notify status. + */ +void BLECharacteristic::setSubscribe(struct ble_gap_event *event) { + ble_gap_conn_desc desc; + if (ble_gap_conn_find(event->subscribe.conn_handle, &desc) != 0) { + return; + } + + uint16_t subVal = 0; + if (event->subscribe.cur_notify > 0 && (m_properties & BLECharacteristic::PROPERTY_NOTIFY)) { + subVal |= NIMBLE_SUB_NOTIFY; + } + if (event->subscribe.cur_indicate && (m_properties & BLECharacteristic::PROPERTY_INDICATE)) { + subVal |= NIMBLE_SUB_INDICATE; + } + + log_i("New subscribe value for conn: %d val: %d", event->subscribe.conn_handle, subVal); + + if (!event->subscribe.cur_indicate && event->subscribe.prev_indicate) { + BLEDevice::getServer()->clearIndicateWait(event->subscribe.conn_handle); + } + + auto it = m_subscribedVec.begin(); + for (; it != m_subscribedVec.end(); ++it) { + if ((*it).first == event->subscribe.conn_handle) { + break; + } + } + + if (subVal > 0) { + if (it == m_subscribedVec.end()) { + m_subscribedVec.push_back({event->subscribe.conn_handle, subVal}); + } else { + (*it).second = subVal; + } + } else if (it != m_subscribedVec.end()) { + m_subscribedVec.erase(it); + } + + m_pCallbacks->onSubscribe(this, &desc, subVal); +} + +/** + * @brief Send a notify. + * A notification is a transmission of up to the first 20 bytes of the characteristic value. An notification + * will not block; it is a fire and forget. + * @return N/A. + */ +void BLECharacteristic::notify(bool is_notification) { + log_v(">> notify: length: %d", m_value.getValue().length()); + + assert(getService() != nullptr); + assert(getService()->getServer() != nullptr); + + int rc = 0; + m_pCallbacks->onNotify(this); // Invoke the notify callback. + + // GeneralUtils::hexDump() doesn't output anything if the log level is not + // "VERBOSE". Additionally, it is very CPU intensive, even when it doesn't + // output anything! So it is much better to *not* call it at all if not needed. + // In a simple program which calls BLECharacteristic::notify() every 50 ms, + // the performance gain of this little optimization is 37% in release mode + // (-O3) and 57% in debug mode. + // Of course, the "#if ARDUHAL_LOG_LEVEL >= ARDUHAL_LOG_LEVEL_VERBOSE" guard + // could also be put inside the GeneralUtils::hexDump() function itself. But + // it's better to put it here also, as it is clearer (indicating a verbose log + // thing) and it allows to remove the "m_value.getValue().c_str()" call, which + // is, in itself, quite CPU intensive. +#if ARDUHAL_LOG_LEVEL >= ARDUHAL_LOG_LEVEL_VERBOSE + GeneralUtils::hexDump((uint8_t *)m_value.getValue().c_str(), m_value.getValue().length()); +#endif + + if (getService()->getServer()->getConnectedCount() == 0) { + log_v("<< notify: No connected clients."); + m_pCallbacks->onStatus(this, BLECharacteristicCallbacks::Status::ERROR_NO_CLIENT, 0); + return; + } + + if (m_subscribedVec.size() == 0) { + log_v("<< notify: No clients subscribed."); + m_pCallbacks->onStatus(this, BLECharacteristicCallbacks::Status::ERROR_NO_SUBSCRIBER, 0); + return; + } + + if (is_notification) { + if (!(m_properties & BLECharacteristic::PROPERTY_NOTIFY)) { + log_v("<< notifications disabled; ignoring"); + m_pCallbacks->onStatus(this, BLECharacteristicCallbacks::Status::ERROR_NOTIFY_DISABLED, 0); // Invoke the notify callback. + return; + } + } else { + if (!(m_properties & BLECharacteristic::PROPERTY_INDICATE)) { + log_v("<< indications disabled; ignoring"); + m_pCallbacks->onStatus(this, BLECharacteristicCallbacks::Status::ERROR_INDICATE_DISABLED, 0); // Invoke the notify callback. + return; + } + } + + bool reqSec = (m_properties & BLE_GATT_CHR_F_READ_AUTHEN) || (m_properties & BLE_GATT_CHR_F_READ_AUTHOR) || (m_properties & BLE_GATT_CHR_F_READ_ENC); + + for (auto &myPair : m_subscribedVec) { + uint16_t _mtu = getService()->getServer()->getPeerMTU(myPair.first); + + // check if connected and subscribed + if (_mtu == 0 || myPair.second == 0) { + continue; + } + + if (reqSec) { + struct ble_gap_conn_desc desc; + rc = ble_gap_conn_find(myPair.first, &desc); + if (rc != 0 || !desc.sec_state.encrypted) { + continue; + } + } + + String value = getValue(); + size_t length = value.length(); + + if (length > _mtu - 3) { + log_w("- Truncating to %d bytes (maximum notify size)", _mtu - 3); + } + + if (is_notification && (!(myPair.second & NIMBLE_SUB_NOTIFY))) { + log_w("Sending notification to client subscribed to indications, sending indication instead"); + is_notification = false; + } + + if (!is_notification && (!(myPair.second & NIMBLE_SUB_INDICATE))) { + log_w("Sending indication to client subscribed to notification, sending notification instead"); + is_notification = true; + } + + if (!is_notification) { // is indication + m_semaphoreConfEvt.take("indicate"); + } + + // don't create the m_buf until we are sure to send the data or else + // we could be allocating a buffer that doesn't get released. + // We also must create it in each loop iteration because it is consumed with each host call. + os_mbuf *om = ble_hs_mbuf_from_flat((uint8_t *)value.c_str(), length); + + if (!is_notification && (m_properties & BLECharacteristic::PROPERTY_INDICATE)) { + if (!BLEDevice::getServer()->setIndicateWait(myPair.first)) { + log_e("prior Indication in progress"); + os_mbuf_free_chain(om); + return; + } + + rc = ble_gatts_indicate_custom(myPair.first, m_handle, om); + if (rc != 0) { + BLEDevice::getServer()->clearIndicateWait(myPair.first); + } + } else { + rc = ble_gatts_notify_custom(myPair.first, m_handle, om); + } + + if (rc != 0) { + log_e("<< ble_gatts_%s_custom: rc=%d %s", is_notification ? "notify" : "indicate", rc, GeneralUtils::errorToString(rc)); + m_semaphoreConfEvt.give(); + m_pCallbacks->onStatus(this, BLECharacteristicCallbacks::Status::ERROR_GATT, rc); // Invoke the notify callback. + return; + } + + if (!is_notification) { // is indication + if (!m_semaphoreConfEvt.timedWait("indicate", indicationTimeout)) { + m_pCallbacks->onStatus(this, BLECharacteristicCallbacks::Status::ERROR_INDICATE_TIMEOUT, 0); // Invoke the notify callback. + } else { + auto code = m_semaphoreConfEvt.value(); + if (code == ESP_OK) { + m_pCallbacks->onStatus(this, BLECharacteristicCallbacks::Status::SUCCESS_INDICATE, code); // Invoke the notify callback. + } else { + m_pCallbacks->onStatus(this, BLECharacteristicCallbacks::Status::ERROR_INDICATE_FAILURE, code); + } + } + } else { + m_pCallbacks->onStatus(this, BLECharacteristicCallbacks::Status::SUCCESS_NOTIFY, 0); // Invoke the notify callback. + } + } + log_v("<< notify"); +} // Notify + +void BLECharacteristicCallbacks::onRead(BLECharacteristic *pCharacteristic, ble_gap_conn_desc *desc) { + onRead(pCharacteristic); +} // onRead + +void BLECharacteristicCallbacks::onWrite(BLECharacteristic *pCharacteristic, ble_gap_conn_desc *desc) { + onWrite(pCharacteristic); +} // onWrite + +void BLECharacteristicCallbacks::onSubscribe(BLECharacteristic *pCharacteristic, ble_gap_conn_desc *desc, uint16_t subValue) { + log_d(">> onSubscribe: default"); + log_d("<< onSubscribe"); +} // onSubscribe + +#endif /* CONFIG_NIMBLE_ENABLED */ + +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ diff --git a/libraries/BLE/src/BLECharacteristic.h b/libraries/BLE/src/BLECharacteristic.h index 29f105868fd..27df5a30c3e 100644 --- a/libraries/BLE/src/BLECharacteristic.h +++ b/libraries/BLE/src/BLECharacteristic.h @@ -3,6 +3,10 @@ * * Created on: Jun 22, 2017 * Author: kolban + * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on kolban's and h2zero's work) + * Description: Added support for NimBLE */ #ifndef COMPONENTS_CPP_UTILS_BLECHARACTERISTIC_H_ @@ -11,15 +15,59 @@ #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) + +/*************************************************************************** + * Common includes * + ***************************************************************************/ + #include #include #include "BLEUUID.h" -#include -#include #include "BLEDescriptor.h" #include "BLEValue.h" #include "RTOS.h" +#include "BLEUtils.h" + +/*************************************************************************** + * Bluedroid includes * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) +#include +#include +#endif + +/*************************************************************************** + * NimBLE includes and definitions * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) +#include +#include +#include +#include "BLEConnInfo.h" +#define ESP_GATT_MAX_ATTR_LEN BLE_ATT_ATTR_MAX_LEN +#define ESP_GATT_CHAR_PROP_BIT_READ BLE_GATT_CHR_PROP_READ +#define ESP_GATT_CHAR_PROP_BIT_WRITE BLE_GATT_CHR_PROP_WRITE +#define ESP_GATT_CHAR_PROP_BIT_WRITE_NR BLE_GATT_CHR_PROP_WRITE_NO_RSP +#define ESP_GATT_CHAR_PROP_BIT_BROADCAST BLE_GATT_CHR_PROP_BROADCAST +#define ESP_GATT_CHAR_PROP_BIT_NOTIFY BLE_GATT_CHR_PROP_NOTIFY +#define ESP_GATT_CHAR_PROP_BIT_INDICATE BLE_GATT_CHR_PROP_INDICATE +#endif + +/*************************************************************************** + * NimBLE types * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) +typedef uint16_t esp_gatt_char_prop_t; +typedef uint8_t esp_gatt_perm_t; +#endif + +/*************************************************************************** + * Forward declarations * + ***************************************************************************/ class BLEService; class BLEDescriptor; @@ -30,6 +78,10 @@ class BLECharacteristicCallbacks; */ class BLEDescriptorMap { public: + /*************************************************************************** + * Common public declarations * + ***************************************************************************/ + void setByUUID(const char *uuid, BLEDescriptor *pDescriptor); void setByUUID(BLEUUID uuid, BLEDescriptor *pDescriptor); void setByHandle(uint16_t handle, BLEDescriptor *pDescriptor); @@ -37,11 +89,32 @@ class BLEDescriptorMap { BLEDescriptor *getByUUID(BLEUUID uuid); BLEDescriptor *getByHandle(uint16_t handle); String toString(); - void handleGATTServerEvent(esp_gatts_cb_event_t event, esp_gatt_if_t gatts_if, esp_ble_gatts_cb_param_t *param); BLEDescriptor *getFirst(); BLEDescriptor *getNext(); + int getRegisteredDescriptorCount(); + void removeDescriptor(BLEDescriptor *pDescriptor); + + /*************************************************************************** + * Bluedroid public declarations * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) + void handleGATTServerEvent(esp_gatts_cb_event_t event, esp_gatt_if_t gatts_if, esp_ble_gatts_cb_param_t *param); +#endif + + /*************************************************************************** + * NimBLE public declarations * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) + void handleGATTServerEvent(uint16_t conn_handle, uint16_t attr_handle, ble_gatt_access_ctxt *ctxt, void *arg); +#endif private: + /*************************************************************************** + * Common private properties * + ***************************************************************************/ + std::map m_uuidMap; std::map m_handleMap; std::map::iterator m_iterator; @@ -55,6 +128,48 @@ class BLEDescriptorMap { */ class BLECharacteristic { public: + /*************************************************************************** + * Common properties * + ***************************************************************************/ + + static const uint32_t indicationTimeout = 1000; + + /*************************************************************************** + * Bluedroid public properties * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) + static const uint32_t PROPERTY_READ = 1 << 0; + static const uint32_t PROPERTY_WRITE = 1 << 1; + static const uint32_t PROPERTY_NOTIFY = 1 << 2; + static const uint32_t PROPERTY_BROADCAST = 1 << 3; + static const uint32_t PROPERTY_INDICATE = 1 << 4; + static const uint32_t PROPERTY_WRITE_NR = 1 << 5; +#endif + + /*************************************************************************** + * NimBLE public properties * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) + static const uint32_t PROPERTY_READ = BLE_GATT_CHR_F_READ; + static const uint32_t PROPERTY_READ_ENC = BLE_GATT_CHR_F_READ_ENC; + static const uint32_t PROPERTY_READ_AUTHEN = BLE_GATT_CHR_F_READ_AUTHEN; + static const uint32_t PROPERTY_READ_AUTHOR = BLE_GATT_CHR_F_READ_AUTHOR; + static const uint32_t PROPERTY_WRITE = BLE_GATT_CHR_F_WRITE; + static const uint32_t PROPERTY_WRITE_NR = BLE_GATT_CHR_F_WRITE_NO_RSP; + static const uint32_t PROPERTY_WRITE_ENC = BLE_GATT_CHR_F_WRITE_ENC; + static const uint32_t PROPERTY_WRITE_AUTHEN = BLE_GATT_CHR_F_WRITE_AUTHEN; + static const uint32_t PROPERTY_WRITE_AUTHOR = BLE_GATT_CHR_F_WRITE_AUTHOR; + static const uint32_t PROPERTY_BROADCAST = BLE_GATT_CHR_F_BROADCAST; + static const uint32_t PROPERTY_NOTIFY = BLE_GATT_CHR_F_NOTIFY; + static const uint32_t PROPERTY_INDICATE = BLE_GATT_CHR_F_INDICATE; +#endif + + /*************************************************************************** + * Common public declarations * + ***************************************************************************/ + BLECharacteristic(const char *uuid, uint32_t properties = 0); BLECharacteristic(BLEUUID uuid, uint32_t properties = 0); virtual ~BLECharacteristic(); @@ -66,14 +181,9 @@ class BLECharacteristic { String getValue(); uint8_t *getData(); size_t getLength(); - void indicate(); void notify(bool is_notification = true); - void setBroadcastProperty(bool value); void setCallbacks(BLECharacteristicCallbacks *pCallbacks); - void setIndicateProperty(bool value); - void setNotifyProperty(bool value); - void setReadProperty(bool value); void setValue(uint8_t *data, size_t size); void setValue(String value); void setValue(uint16_t &data16); @@ -81,20 +191,16 @@ class BLECharacteristic { void setValue(int &data32); void setValue(float &data32); void setValue(double &data64); - void setWriteProperty(bool value); - void setWriteNoResponseProperty(bool value); String toString(); uint16_t getHandle(); - void setAccessPermissions(esp_gatt_perm_t perm); - - static const uint32_t PROPERTY_READ = 1 << 0; - static const uint32_t PROPERTY_WRITE = 1 << 1; - static const uint32_t PROPERTY_NOTIFY = 1 << 2; - static const uint32_t PROPERTY_BROADCAST = 1 << 3; - static const uint32_t PROPERTY_INDICATE = 1 << 4; - static const uint32_t PROPERTY_WRITE_NR = 1 << 5; - - static const uint32_t indicationTimeout = 1000; + void setAccessPermissions(uint8_t perm); + esp_gatt_char_prop_t getProperties(); + void setReadProperty(bool value); + void setWriteProperty(bool value); + void setNotifyProperty(bool value); + void setBroadcastProperty(bool value); + void setIndicateProperty(bool value); + void setWriteNoResponseProperty(bool value); private: friend class BLEServer; @@ -102,6 +208,10 @@ class BLECharacteristic { friend class BLEDescriptor; friend class BLECharacteristicMap; + /*************************************************************************** + * Common private properties * + ***************************************************************************/ + BLEUUID m_bleUUID; BLEDescriptorMap m_descriptorMap; uint16_t m_handle; @@ -109,18 +219,53 @@ class BLECharacteristic { BLECharacteristicCallbacks *m_pCallbacks; BLEService *m_pService; BLEValue m_value; + bool m_writeEvt = false; + FreeRTOS::Semaphore m_semaphoreConfEvt = FreeRTOS::Semaphore("ConfEvt"); + FreeRTOS::Semaphore m_semaphoreSetValue = FreeRTOS::Semaphore("SetValue"); + + /*************************************************************************** + * Bluedroid private properties * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) + FreeRTOS::Semaphore m_semaphoreCreateEvt = FreeRTOS::Semaphore("CreateEvt"); esp_gatt_perm_t m_permissions = ESP_GATT_PERM_READ | ESP_GATT_PERM_WRITE; - bool m_writeEvt = false; // If we have started a long write, this tells the commit code that we were the target +#endif - void handleGATTServerEvent(esp_gatts_cb_event_t event, esp_gatt_if_t gatts_if, esp_ble_gatts_cb_param_t *param); + /*************************************************************************** + * NimBLE private properties * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) + portMUX_TYPE m_readMux; + uint8_t m_removed; + std::vector> m_subscribedVec; +#endif + + /*************************************************************************** + * Common private declarations * + ***************************************************************************/ void executeCreate(BLEService *pService); - esp_gatt_char_prop_t getProperties(); BLEService *getService(); void setHandle(uint16_t handle); - FreeRTOS::Semaphore m_semaphoreCreateEvt = FreeRTOS::Semaphore("CreateEvt"); - FreeRTOS::Semaphore m_semaphoreConfEvt = FreeRTOS::Semaphore("ConfEvt"); - FreeRTOS::Semaphore m_semaphoreSetValue = FreeRTOS::Semaphore("SetValue"); + + /*************************************************************************** + * Bluedroid private declarations * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) + void handleGATTServerEvent(esp_gatts_cb_event_t event, esp_gatt_if_t gatts_if, esp_ble_gatts_cb_param_t *param); +#endif + + /*************************************************************************** + * NimBLE private declarations * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) + void setSubscribe(struct ble_gap_event *event); + static int handleGATTServerEvent(uint16_t conn_handle, uint16_t attr_handle, struct ble_gatt_access_ctxt *ctxt, void *arg); +#endif }; // BLECharacteristic /** @@ -132,6 +277,10 @@ class BLECharacteristic { */ class BLECharacteristicCallbacks { public: + /*************************************************************************** + * Common public types * + ***************************************************************************/ + typedef enum { SUCCESS_INDICATE, SUCCESS_NOTIFY, @@ -139,51 +288,41 @@ class BLECharacteristicCallbacks { ERROR_NOTIFY_DISABLED, ERROR_GATT, ERROR_NO_CLIENT, + ERROR_NO_SUBSCRIBER, ERROR_INDICATE_TIMEOUT, ERROR_INDICATE_FAILURE } Status; - virtual ~BLECharacteristicCallbacks(); + /*************************************************************************** + * Common public declarations * + ***************************************************************************/ - /** - * @brief Callback function to support a read request. - * @param [in] pCharacteristic The characteristic that is the source of the event. - * @param [in] param The BLE GATTS param. Use param->read. - */ - virtual void onRead(BLECharacteristic *pCharacteristic, esp_ble_gatts_cb_param_t *param); - /** - * @brief DEPRECATED! Callback function to support a read request. Called only if onRead(,) is not overridden - * @param [in] pCharacteristic The characteristic that is the source of the event. - */ + virtual ~BLECharacteristicCallbacks(); virtual void onRead(BLECharacteristic *pCharacteristic); + virtual void onWrite(BLECharacteristic *pCharacteristic); + virtual void onNotify(BLECharacteristic *pCharacteristic); + virtual void onStatus(BLECharacteristic *pCharacteristic, Status s, uint32_t code); - /** - * @brief Callback function to support a write request. - * @param [in] pCharacteristic The characteristic that is the source of the event. - * @param [in] param The BLE GATTS param. Use param->write. - */ + /*************************************************************************** + * Bluedroid public declarations * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) + virtual void onRead(BLECharacteristic *pCharacteristic, esp_ble_gatts_cb_param_t *param); virtual void onWrite(BLECharacteristic *pCharacteristic, esp_ble_gatts_cb_param_t *param); - /** - * @brief DEPRECATED! Callback function to support a write request. Called only if onWrite(,) is not overridden. - * @param [in] pCharacteristic The characteristic that is the source of the event. - */ - virtual void onWrite(BLECharacteristic *pCharacteristic); +#endif - /** - * @brief Callback function to support a Notify request. - * @param [in] pCharacteristic The characteristic that is the source of the event. - */ - virtual void onNotify(BLECharacteristic *pCharacteristic); + /*************************************************************************** + * NimBLE public declarations * + ***************************************************************************/ - /** - * @brief Callback function to support a Notify/Indicate Status report. - * @param [in] pCharacteristic The characteristic that is the source of the event. - * @param [in] s Status of the notification/indication - * @param [in] code Additional code of underlying errors - */ - virtual void onStatus(BLECharacteristic *pCharacteristic, Status s, uint32_t code); +#if defined(CONFIG_NIMBLE_ENABLED) + virtual void onRead(BLECharacteristic *pCharacteristic, ble_gap_conn_desc *desc); + virtual void onWrite(BLECharacteristic *pCharacteristic, ble_gap_conn_desc *desc); + virtual void onSubscribe(BLECharacteristic *pCharacteristic, ble_gap_conn_desc *desc, uint16_t subValue); +#endif }; -#endif /* CONFIG_BLUEDROID_ENABLED */ +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ #endif /* COMPONENTS_CPP_UTILS_BLECHARACTERISTIC_H_ */ diff --git a/libraries/BLE/src/BLECharacteristicMap.cpp b/libraries/BLE/src/BLECharacteristicMap.cpp index 6f2c0bb1154..ec3588bcde4 100644 --- a/libraries/BLE/src/BLECharacteristicMap.cpp +++ b/libraries/BLE/src/BLECharacteristicMap.cpp @@ -3,19 +3,34 @@ * * Created on: Jun 22, 2017 * Author: kolban + * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on kolban's and h2zero's work) + * Description: Added support for NimBLE */ + #include "soc/soc_caps.h" #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) + +/*************************************************************************** + * Common includes * + ***************************************************************************/ + #include #include #include "BLEService.h" +#include "BLEUtils.h" #ifdef ARDUINO_ARCH_ESP32 #include "esp32-hal-log.h" #endif +/*************************************************************************** + * Common functions * + ***************************************************************************/ + /** * @brief Return the characteristic by handle. * @param [in] handle The handle to look up the characteristic. @@ -77,17 +92,22 @@ BLECharacteristic *BLECharacteristicMap::getNext() { } // getNext /** - * @brief Pass the GATT server event onwards to each of the characteristics found in the mapping - * @param [in] event - * @param [in] gatts_if - * @param [in] param + * @brief Get the number of registered characteristics. + * @return The number of registered characteristics. */ -void BLECharacteristicMap::handleGATTServerEvent(esp_gatts_cb_event_t event, esp_gatt_if_t gatts_if, esp_ble_gatts_cb_param_t *param) { - // Invoke the handler for every Service we have. - for (auto &myPair : m_uuidMap) { - myPair.first->handleGATTServerEvent(event, gatts_if, param); - } -} // handleGATTServerEvent +int BLECharacteristicMap::getRegisteredCharacteristicCount() { + return m_uuidMap.size(); +} // getRegisteredCharacteristicCount + +/** + * @brief Removes characteristic from maps. + * @param [in] characteristic The characteristic to remove. + * @return N/A. + */ +void BLECharacteristicMap::removeCharacteristic(BLECharacteristic *characteristic) { + m_handleMap.erase(characteristic->getHandle()); + m_uuidMap.erase(characteristic); +} // removeCharacteristic /** * @brief Set the characteristic by handle. @@ -130,5 +150,37 @@ String BLECharacteristicMap::toString() { return res; } // toString -#endif /* CONFIG_BLUEDROID_ENABLED */ +/*************************************************************************** + * Bluedroid functions * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) +/** + * @brief Pass the GATT server event onwards to each of the characteristics found in the mapping + * @param [in] event + * @param [in] gatts_if + * @param [in] param + */ +void BLECharacteristicMap::handleGATTServerEvent(esp_gatts_cb_event_t event, esp_gatt_if_t gatts_if, esp_ble_gatts_cb_param_t *param) { + // Invoke the handler for every Service we have. + for (auto &myPair : m_uuidMap) { + myPair.first->handleGATTServerEvent(event, gatts_if, param); + } +} // handleGATTServerEvent +#endif // CONFIG_BLUEDROID_ENABLED + +/*************************************************************************** + * NimBLE functions * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) +void BLECharacteristicMap::handleGATTServerEvent(uint16_t conn_handle, uint16_t attr_handle, ble_gatt_access_ctxt *ctxt, void *arg) { + // Invoke the handler for every Service we have. + for (auto &myPair : m_uuidMap) { + myPair.first->handleGATTServerEvent(conn_handle, attr_handle, ctxt, arg); + } +} +#endif // CONFIG_NIMBLE_ENABLED + +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ diff --git a/libraries/BLE/src/BLEClient.cpp b/libraries/BLE/src/BLEClient.cpp index 29fa0fbc140..c101993d002 100644 --- a/libraries/BLE/src/BLEClient.cpp +++ b/libraries/BLE/src/BLEClient.cpp @@ -3,17 +3,23 @@ * * Created on: Mar 22, 2017 * Author: kolban + * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on kolban's and h2zero's work) + * Description: Added support for NimBLE */ + #include "soc/soc_caps.h" #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) + +/*************************************************************************** + * Common includes * + ***************************************************************************/ + #include -#include -#include -#include -#include // ESP32 BLE #include "BLEClient.h" #include "BLEUtils.h" #include "BLEService.h" @@ -24,6 +30,39 @@ #include "BLEDevice.h" #include "esp32-hal-log.h" +/*************************************************************************** + * Bluedroid includes * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) +#include +#include +#include +#include +#endif + +/*************************************************************************** + * NimBLE includes * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) +#include +#include +#include +#include +#include +#endif + +/*************************************************************************** + * Common global variables * + ***************************************************************************/ + +static BLEClientCallbacks defaultCallbacks; + +/*************************************************************************** + * Common functions * + ***************************************************************************/ + /* * Design * ------ @@ -46,11 +85,30 @@ */ BLEClient::BLEClient() { - m_pClientCallbacks = nullptr; + m_pClientCallbacks = &defaultCallbacks; m_conn_id = ESP_GATT_IF_NONE; - m_gattc_if = ESP_GATT_IF_NONE; m_haveServices = false; m_isConnected = false; // Initially, we are flagged as not connected. + +#if defined(CONFIG_BLUEDROID_ENABLED) + m_gattc_if = ESP_GATT_IF_NONE; +#endif + +#if defined(CONFIG_NIMBLE_ENABLED) + m_connectTimeout = 30000; + m_pTaskData = nullptr; + m_lastErr = 0; + m_terminateFailCount = 0; + + m_pConnParams.scan_itvl = 16; // Scan interval in 0.625ms units (NimBLE Default) + m_pConnParams.scan_window = 16; // Scan window in 0.625ms units (NimBLE Default) + m_pConnParams.itvl_min = BLE_GAP_INITIAL_CONN_ITVL_MIN; // min_int = 0x10*1.25ms = 20ms + m_pConnParams.itvl_max = BLE_GAP_INITIAL_CONN_ITVL_MAX; // max_int = 0x20*1.25ms = 40ms + m_pConnParams.latency = BLE_GAP_INITIAL_CONN_LATENCY; // number of packets allowed to skip (extends max interval) + m_pConnParams.supervision_timeout = BLE_GAP_INITIAL_SUPERVISION_TIMEOUT; // timeout = 400*10ms = 4000ms + m_pConnParams.min_ce_len = BLE_GAP_INITIAL_CONN_MIN_CE_LEN; // Minimum length of connection event in 0.625ms units + m_pConnParams.max_ce_len = BLE_GAP_INITIAL_CONN_MAX_CE_LEN; // Maximum length of connection event in 0.625ms units +#endif } // BLEClient /** @@ -86,7 +144,7 @@ void BLEClient::clearServices() { */ bool BLEClient::connect(BLEAdvertisedDevice *device) { BLEAddress address = device->getAddress(); - esp_ble_addr_type_t type = device->getAddressType(); + uint8_t type = device->getAddressType(); return connect(address, type); } @@ -95,10 +153,262 @@ bool BLEClient::connect(BLEAdvertisedDevice *device) { */ bool BLEClient::connectTimeout(BLEAdvertisedDevice *device, uint32_t timeoutMs) { BLEAddress address = device->getAddress(); - esp_ble_addr_type_t type = device->getAddressType(); + uint8_t type = device->getAddressType(); return connect(address, type, timeoutMs); } +esp_gatt_if_t BLEClient::getGattcIf() { +#if defined(CONFIG_BLUEDROID_ENABLED) + return m_gattc_if; +#endif + +#if defined(CONFIG_NIMBLE_ENABLED) + log_e("NimBLE does not support getGattcIf()"); + return ESP_GATT_IF_NONE; +#endif +} // getGattcIf + +/** + * @brief Initiate a secure connection (pair/bond) with the server.\n + * Called automatically when a characteristic or descriptor requires encryption or authentication to access it. + * @return True on success. + */ +bool BLEClient::secureConnection() { +#if defined(CONFIG_BLUEDROID_ENABLED) + log_i("secureConnection() does not need to be called for Bluedroid"); + return true; +#endif + +#if defined(CONFIG_NIMBLE_ENABLED) + int retryCount = 1; + BLETaskData taskData(const_cast(this), BLE_HS_ENOTCONN); + m_pTaskData = &taskData; + + do { + if (BLESecurity::startSecurity(m_conn_id)) { + BLEUtils::taskWait(taskData, BLE_NPL_TIME_FOREVER); + } + + } while (taskData.m_flags == (BLE_HS_ERR_HCI_BASE + BLE_ERR_PINKEY_MISSING) && retryCount--); + + m_pTaskData = nullptr; + + if (taskData.m_flags == 0) { + log_d("<< secureConnection: success"); + return true; + } + + m_lastErr = taskData.m_flags; + log_e("secureConnection: failed rc=%d", taskData.m_flags); + return false; +#endif +} // secureConnection + +uint16_t BLEClient::getConnId() { + return m_conn_id; +} // getConnId + +/** + * @brief Retrieve the address of the peer. + * + * Returns the Bluetooth device address of the %BLE peer to which this client is connected. + */ +BLEAddress BLEClient::getPeerAddress() { + return m_peerAddress; +} // getAddress + +/** + * @brief Ask the BLE server for the RSSI value. + * @return The RSSI value. + */ +int BLEClient::getRssi() { + log_v(">> getRssi()"); + if (!isConnected()) { + log_v("<< getRssi(): Not connected"); + return 0; + } + +#if defined(CONFIG_BLUEDROID_ENABLED) + // We make the API call to read the RSSI value which is an asynchronous operation. We expect to receive + // an ESP_GAP_BLE_READ_RSSI_COMPLETE_EVT to indicate completion. + // + m_semaphoreRssiCmplEvt.take("getRssi"); + esp_err_t rc = ::esp_ble_gap_read_rssi(getPeerAddress().getNative()); + if (rc != ESP_OK) { + log_e("<< getRssi: esp_ble_gap_read_rssi: rc=%d %s", rc, GeneralUtils::errorToString(rc)); + return 0; + } + int rssiValue = m_semaphoreRssiCmplEvt.wait("getRssi"); +#endif // CONFIG_BLUEDROID_ENABLED + +#if defined(CONFIG_NIMBLE_ENABLED) + int8_t rssiValue = 0; + int rc = ble_gap_conn_rssi(m_conn_id, &rssiValue); + if (rc != 0) { + log_e("<< getRssi: ble_gap_conn_rssi: rc=%d %s", rc, BLEUtils::returnCodeToString(rc)); + return 0; + } +#endif // CONFIG_BLUEDROID_ENABLED + log_v("<< getRssi(): %d", rssiValue); + return rssiValue; +} // getRssi + +/** + * @brief Get the service BLE Remote Service instance corresponding to the uuid. + * @param [in] uuid The UUID of the service being sought. + * @return A reference to the Service or nullptr if don't know about it. + */ +BLERemoteService *BLEClient::getService(const char *uuid) { + return getService(BLEUUID(uuid)); +} // getService + +/** + * @brief Get the service object corresponding to the uuid. + * @param [in] uuid The UUID of the service being sought. + * @return A reference to the Service or nullptr if don't know about it. + * @throws BLEUuidNotFound + */ +BLERemoteService *BLEClient::getService(BLEUUID uuid) { + log_v(">> getService: uuid: %s", uuid.toString().c_str()); + // Design + // ------ + // We wish to retrieve the service given its UUID. It is possible that we have not yet asked the + // device what services it has in which case we have nothing to match against. If we have not + // asked the device about its services, then we do that now. Once we get the results we can then + // examine the services map to see if it has the service we are looking for. + if (!m_haveServices) { + getServices(); + } + std::string uuidStr = uuid.toString().c_str(); + for (auto &myPair : m_servicesMap) { + if (myPair.first == uuidStr) { + log_v("<< getService: found the service with uuid: %s", uuid.toString().c_str()); + return myPair.second; + } + } // End of each of the services. + log_v("<< getService: not found"); + return nullptr; +} // getService + +/** + * @brief Get the value of a specific characteristic associated with a specific service. + * @param [in] serviceUUID The service that owns the characteristic. + * @param [in] characteristicUUID The characteristic whose value we wish to read. + * @throws BLEUuidNotFound + */ +String BLEClient::getValue(BLEUUID serviceUUID, BLEUUID characteristicUUID) { + log_v(">> getValue: serviceUUID: %s, characteristicUUID: %s", serviceUUID.toString().c_str(), characteristicUUID.toString().c_str()); + String ret = getService(serviceUUID)->getCharacteristic(characteristicUUID)->readValue(); + log_v("<> setValue: serviceUUID: %s, characteristicUUID: %s", serviceUUID.toString().c_str(), characteristicUUID.toString().c_str()); + getService(serviceUUID)->getCharacteristic(characteristicUUID)->writeValue(value); + log_v("<< setValue"); +} // setValue + +uint16_t BLEClient::getMTU() { +#ifdef CONFIG_BLUEDROID_ENABLED + return m_mtu; +#endif + +#if defined(CONFIG_NIMBLE_ENABLED) + return ble_att_mtu(m_conn_id); +#endif +} + +/** + @brief Set the local and remote MTU size. + Should be called once after client connects if MTU size needs to be changed. + @return bool indicating if MTU was successfully set locally and on remote. +*/ +bool BLEClient::setMTU(uint16_t mtu) { + log_v(">> setMTU: %d", mtu); + esp_err_t err = ESP_OK; + +#ifdef CONFIG_BLUEDROID_ENABLED + err = esp_ble_gatt_set_local_mtu(mtu); //First must set local MTU value. +#endif + +#if defined(CONFIG_NIMBLE_ENABLED) + err = ble_att_set_preferred_mtu(mtu); +#endif + + if (err == ESP_OK) { +#ifdef CONFIG_BLUEDROID_ENABLED + err = esp_ble_gattc_send_mtu_req(m_gattc_if, m_conn_id); //Once local is set successfully set remote size +#endif + +#if defined(CONFIG_NIMBLE_ENABLED) + //err = ble_gattc_exchange_mtu(m_conn_id, nullptr, nullptr); +#endif + + if (err != ESP_OK) { + log_e("Error setting send MTU request MTU: %d err=%d", mtu, err); + return false; + } + } else { + log_e("can't set local mtu value: %d", mtu); + return false; + } + log_v("<< setMTU"); + + m_mtu = mtu; //successfully changed + + return true; +} + +/** + * @brief Return a string representation of this client. + * @return A string representation of this client. + */ +String BLEClient::toString() { + String res = "peer address: " + m_peerAddress.toString(); + res += "\nServices:\n"; + for (auto &myPair : m_servicesMap) { + res += myPair.second->toString() + "\n"; + // myPair.second is the value + } + return res; +} // toString + +void BLEClientCallbacks::onConnect(BLEClient *pClient) { + log_d("BLEClientCallbacks", "onConnect: default"); +} + +void BLEClientCallbacks::onDisconnect(BLEClient *pClient) { + log_d("BLEClientCallbacks", "onDisconnect: default"); +} + +/*************************************************************************** + * Bluedroid functions * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) + /** * @brief Connect to the partner (BLE Server). * @param [in] address The address of the partner. @@ -106,7 +416,7 @@ bool BLEClient::connectTimeout(BLEAdvertisedDevice *device, uint32_t timeoutMs) * @param [in] timeoutMs The number of milliseconds to wait for the connection to complete. * @return True on success. */ -bool BLEClient::connect(BLEAddress address, esp_ble_addr_type_t type, uint32_t timeoutMs) { +bool BLEClient::connect(BLEAddress address, uint8_t type, uint32_t timeoutMs) { log_v(">> connect(%s)", address.toString().c_str()); // We need the connection handle that we get from registering the application. We register the app @@ -141,9 +451,9 @@ bool BLEClient::connect(BLEAddress address, esp_ble_addr_type_t type, uint32_t t m_semaphoreOpenEvt.take("connect"); errRc = ::esp_ble_gattc_open( m_gattc_if, - *getPeerAddress().getNative(), // address - type, // Note: This was added on 2018-04-03 when the latest ESP-IDF was detected to have changed the signature. - 1 // direct connection <-- maybe needs to be changed in case of direct indirect connection??? + getPeerAddress().getNative(), // address + (esp_ble_addr_type_t)type, // Note: This was added on 2018-04-03 when the latest ESP-IDF was detected to have changed the signature. + 1 // direct connection <-- maybe needs to be changed in case of direct indirect connection??? ); if (errRc != ESP_OK) { log_e("esp_ble_gattc_open: rc=%d %s", errRc, GeneralUtils::errorToString(errRc)); @@ -165,16 +475,17 @@ bool BLEClient::connect(BLEAddress address, esp_ble_addr_type_t type, uint32_t t /** * @brief Disconnect from the peer. - * @return N/A. + * @return error code from bluedroid, 0 = success. */ -void BLEClient::disconnect() { +int BLEClient::disconnect(uint8_t reason) { log_v(">> disconnect()"); esp_err_t errRc = ::esp_ble_gattc_close(getGattcIf(), getConnId()); if (errRc != ESP_OK) { log_e("esp_ble_gattc_close: rc=%d %s", errRc, GeneralUtils::errorToString(errRc)); - return; + return errRc; } log_v("<< disconnect()"); + return ESP_OK; } // disconnect /** @@ -361,89 +672,11 @@ void BLEClient::gattClientEventHandler(esp_gattc_cb_event_t event, esp_gatt_if_t } // gattClientEventHandler -uint16_t BLEClient::getConnId() { - return m_conn_id; -} // getConnId - -esp_gatt_if_t BLEClient::getGattcIf() { - return m_gattc_if; -} // getGattcIf - -/** - * @brief Retrieve the address of the peer. - * - * Returns the Bluetooth device address of the %BLE peer to which this client is connected. - */ -BLEAddress BLEClient::getPeerAddress() { - return m_peerAddress; -} // getAddress - /** - * @brief Ask the BLE server for the RSSI value. - * @return The RSSI value. - */ -int BLEClient::getRssi() { - log_v(">> getRssi()"); - if (!isConnected()) { - log_v("<< getRssi(): Not connected"); - return 0; - } - // We make the API call to read the RSSI value which is an asynchronous operation. We expect to receive - // an ESP_GAP_BLE_READ_RSSI_COMPLETE_EVT to indicate completion. - // - m_semaphoreRssiCmplEvt.take("getRssi"); - esp_err_t rc = ::esp_ble_gap_read_rssi(*getPeerAddress().getNative()); - if (rc != ESP_OK) { - log_e("<< getRssi: esp_ble_gap_read_rssi: rc=%d %s", rc, GeneralUtils::errorToString(rc)); - return 0; - } - int rssiValue = m_semaphoreRssiCmplEvt.wait("getRssi"); - log_v("<< getRssi(): %d", rssiValue); - return rssiValue; -} // getRssi - -/** - * @brief Get the service BLE Remote Service instance corresponding to the uuid. - * @param [in] uuid The UUID of the service being sought. - * @return A reference to the Service or nullptr if don't know about it. - */ -BLERemoteService *BLEClient::getService(const char *uuid) { - return getService(BLEUUID(uuid)); -} // getService - -/** - * @brief Get the service object corresponding to the uuid. - * @param [in] uuid The UUID of the service being sought. - * @return A reference to the Service or nullptr if don't know about it. - * @throws BLEUuidNotFound - */ -BLERemoteService *BLEClient::getService(BLEUUID uuid) { - log_v(">> getService: uuid: %s", uuid.toString().c_str()); - // Design - // ------ - // We wish to retrieve the service given its UUID. It is possible that we have not yet asked the - // device what services it has in which case we have nothing to match against. If we have not - // asked the device about its services, then we do that now. Once we get the results we can then - // examine the services map to see if it has the service we are looking for. - if (!m_haveServices) { - getServices(); - } - std::string uuidStr = uuid.toString().c_str(); - for (auto &myPair : m_servicesMap) { - if (myPair.first == uuidStr) { - log_v("<< getService: found the service with uuid: %s", uuid.toString().c_str()); - return myPair.second; - } - } // End of each of the services. - log_v("<< getService: not found"); - return nullptr; -} // getService - -/** - * @brief Ask the remote %BLE server for its services. - * A %BLE Server exposes a set of services for its partners. Here we ask the server for its set of - * services and wait until we have received them all. - * @return N/A + * @brief Ask the remote %BLE server for its services. + * A %BLE Server exposes a set of services for its partners. Here we ask the server for its set of + * services and wait until we have received them all. + * @return N/A */ std::map *BLEClient::getServices() { /* @@ -473,19 +706,6 @@ std::map *BLEClient::getServices() { return &m_servicesMap; } // getServices -/** - * @brief Get the value of a specific characteristic associated with a specific service. - * @param [in] serviceUUID The service that owns the characteristic. - * @param [in] characteristicUUID The characteristic whose value we wish to read. - * @throws BLEUuidNotFound - */ -String BLEClient::getValue(BLEUUID serviceUUID, BLEUUID characteristicUUID) { - log_v(">> getValue: serviceUUID: %s, characteristicUUID: %s", serviceUUID.toString().c_str(), characteristicUUID.toString().c_str()); - String ret = getService(serviceUUID)->getCharacteristic(characteristicUUID)->readValue(); - log_v("<> connect(%s)", address.toString().c_str()); + + if (!BLEDevice::m_synced) { + log_d("BLEClient", "Host reset, wait for sync."); + return false; + } + + if (m_conn_id != BLE_HS_CONN_HANDLE_NONE || m_isConnected || m_pTaskData != nullptr) { + log_e("Client busy, connected to %s, id=%d", m_peerAddress.toString().c_str(), getConnId()); + return false; + } + + ble_addr_t peerAddr_t; + memcpy(&peerAddr_t.val, address.getNative(), 6); + peerAddr_t.type = address.getType(); + if (ble_gap_conn_find_by_addr(&peerAddr_t, NULL) == 0) { + log_e("A connection to %s already exists", address.toString().c_str()); + return false; + } + + if (address == BLEAddress("")) { + log_e("Invalid peer address (NULL)"); + return false; + } + + m_appId = BLEDevice::m_appId++; + m_peerAddress = address; + + int rc = 0; + + /* Try to connect the the advertiser. Allow 30 seconds (30000 ms) for + * timeout (default value of m_connectTimeout). + * Loop on BLE_HS_EBUSY if the scan hasn't stopped yet. + */ + do { + rc = ble_gap_connect(BLEDevice::m_ownAddrType, &peerAddr_t, m_connectTimeout, &m_pConnParams, BLEClient::handleGAPEvent, this); + switch (rc) { + case 0: break; + + case BLE_HS_EBUSY: + // Scan was still running, stop it and try again + if (!BLEDevice::getScan()->stop()) { + rc = BLE_HS_EUNKNOWN; + } + break; + + case BLE_HS_EDONE: + // A connection to this device already exists, do not connect twice. + log_e("Already connected to device; addr=%s", m_peerAddress.toString().c_str()); + break; + + case BLE_HS_EALREADY: + // Already attempting to connect to this device, cancel the previous + // attempt and report failure here so we don't get 2 connections. + log_e("Already attempting to connect to %s - canceling", m_peerAddress.toString().c_str()); + ble_gap_conn_cancel(); + break; + + default: log_e("Failed to connect to %s, rc=%d; %s", m_peerAddress.toString().c_str(), rc, BLEUtils::returnCodeToString(rc)); break; + } + + } while (rc == BLE_HS_EBUSY); + + if (rc != 0) { + m_lastErr = rc; + return false; + } + + BLETaskData taskData(this); + m_pTaskData = &taskData; + + // Wait for the connect timeout time +1 second for the connection to complete + if (!BLEUtils::taskWait(taskData, m_connectTimeout + 1000)) { + // If a connection was made but no response from MTU exchange proceed anyway + if (isConnected()) { + taskData.m_flags = 0; + } else { + // workaround; if the controller doesn't cancel the connection at the timeout, cancel it here. + log_e("Connect timeout - canceling"); + ble_gap_conn_cancel(); + taskData.m_flags = BLE_HS_ETIMEOUT; + } + } + + m_pTaskData = nullptr; + rc = taskData.m_flags; + + if (rc != 0) { + log_e("Connection failed; status=%d %s", rc, BLEUtils::returnCodeToString(rc)); + m_lastErr = rc; + return false; + } + + m_isConnected = true; + m_pClientCallbacks->onConnect(this); + + log_i("<< connect()"); + + BLEDevice::addPeerDevice(this, true, m_appId); + // Check if still connected before returning + return isConnected(); +} /** - * @brief Set the value of a specific characteristic associated with a specific service. - * @param [in] serviceUUID The service that owns the characteristic. - * @param [in] characteristicUUID The characteristic whose value we wish to write. - * @throws BLEUuidNotFound + * @brief STATIC Callback for the service discovery API function.\n + * When a service is found or there is none left or there was an error + * the API will call this and report findings. */ -void BLEClient::setValue(BLEUUID serviceUUID, BLEUUID characteristicUUID, String value) { - log_v(">> setValue: serviceUUID: %s, characteristicUUID: %s", serviceUUID.toString().c_str(), characteristicUUID.toString().c_str()); - getService(serviceUUID)->getCharacteristic(characteristicUUID)->writeValue(value); - log_v("<< setValue"); -} // setValue +int BLEClient::serviceDiscoveredCB(uint16_t conn_handle, const struct ble_gatt_error *error, const struct ble_gatt_svc *service, void *arg) { + log_d("Service Discovered >> status: %d handle: %d", error->status, (error->status == 0) ? service->start_handle : -1); -uint16_t BLEClient::getMTU() { - return m_mtu; + BLETaskData *pTaskData = (BLETaskData *)arg; + BLEClient *client = (BLEClient *)pTaskData->m_pInstance; + + if (error->status == BLE_HS_ENOTCONN) { + log_e("<< Service Discovered; Disconnected"); + BLEUtils::taskRelease(*pTaskData, error->status); + return error->status; + } + + // Make sure the service discovery is for this device + if (client->getConnId() != conn_handle) { + return 0; + } + + if (error->status == 0) { + // Found a service - add it to the vector + BLERemoteService *pRemoteService = new BLERemoteService(client, service); + client->m_servicesMap.insert(std::pair(pRemoteService->getUUID().toString().c_str(), pRemoteService)); + client->m_servicesMapByInstID.insert(std::pair(pRemoteService, service->start_handle)); + return 0; + } + + BLEUtils::taskRelease(*pTaskData, error->status); + log_d("<< Service Discovered"); + return error->status; } +std::map *BLEClient::getServices() { + log_v(">> getServices"); + + clearServices(); // Clear any services that may exist. + + if (!isConnected()) { + log_e("Disconnected, could not retrieve services -aborting"); + return &m_servicesMap; + } + + int errRc = 0; + BLETaskData taskData(this); + + errRc = ble_gattc_disc_all_svcs(m_conn_id, BLEClient::serviceDiscoveredCB, &taskData); + + if (errRc != 0) { + log_e("ble_gattc_disc_all_svcs: rc=%d %s", errRc, BLEUtils::returnCodeToString(errRc)); + m_lastErr = errRc; + return &m_servicesMap; + } + + BLEUtils::taskWait(taskData, BLE_NPL_TIME_FOREVER); + errRc = taskData.m_flags; + if (errRc == 0 || errRc == BLE_HS_EDONE) { + // If successful, remember that we now have services. + m_haveServices = m_servicesMap.size() > 0; + log_v("<< getServices"); + return &m_servicesMap; + } + + m_lastErr = errRc; + log_e("Could not retrieve services, rc=%d %s", errRc, BLEUtils::returnCodeToString(errRc)); + return &m_servicesMap; +} // getServices + +int BLEClient::handleGAPEvent(struct ble_gap_event *event, void *arg) { + BLEClient *client = (BLEClient *)arg; + int rc = 0; + BLETaskData *pTaskData = client->m_pTaskData; + + log_d("BLEClient", "Got Client event %s", BLEUtils::gapEventToString(event->type)); + + switch (event->type) { + case BLE_GAP_EVENT_DISCONNECT: + { + rc = event->disconnect.reason; + // If Host reset tell the device now before returning to prevent + // any errors caused by calling host functions before resyncing. + switch (rc) { + case BLE_HS_ECONTROLLER: + case BLE_HS_ETIMEOUT_HCI: + case BLE_HS_ENOTSYNCED: + case BLE_HS_EOS: + log_e("BLEClient", "Disconnect - host reset, rc=%d", rc); + BLEDevice::onReset(rc); + break; + default: + // Check that the event is for this client. + if (client->m_conn_id != event->disconnect.conn.conn_handle) { + return 0; + } + break; + } + + // Remove the device from ignore list so we will scan it again + // BLEDevice::removeIgnored(client->m_peerAddress); + + // No longer connected, clear the connection ID. + client->m_conn_id = BLE_HS_CONN_HANDLE_NONE; + client->m_terminateFailCount = 0; + + // If we received a connected event but did not get established (no PDU) + // then a disconnect event will be sent but we should not send it to the + // app for processing. Instead we will ensure the task is released + // and report the error. + if (!client->m_isConnected) { + break; + } + + log_i("BLEClient", "disconnect; reason=%d, %s", rc, BLEUtils::returnCodeToString(rc)); + + BLEDevice::removePeerDevice(client->m_appId, true); + client->m_isConnected = false; + if (client->m_pClientCallbacks != nullptr) { + client->m_pClientCallbacks->onDisconnect(client); + } + break; + } // BLE_GAP_EVENT_DISCONNECT + + case BLE_GAP_EVENT_CONNECT: + { + // If we aren't waiting for this connection response + // we should drop the connection immediately. + if (client->isConnected() || client->m_pTaskData == nullptr) { + ble_gap_terminate(event->connect.conn_handle, BLE_ERR_REM_USER_CONN_TERM); + return 0; + } + + rc = event->connect.status; + if (rc == 0) { + log_i("BLEClient: Connected event. Handle: %d", event->connect.conn_handle); + + client->m_conn_id = event->connect.conn_handle; + + rc = ble_gattc_exchange_mtu(client->m_conn_id, nullptr, nullptr); + if (rc != 0) { + log_e("BLEClient", "MTU exchange error; rc=%d %s", rc, BLEUtils::returnCodeToString(rc)); + break; + } + + // In the case of a multiconnecting device we ignore this device when + // scanning since we are already connected to it + // BLEDevice::addIgnored(client->m_peerAddress); + } else { + client->m_conn_id = BLE_HS_CONN_HANDLE_NONE; + break; + } + + return 0; + } // BLE_GAP_EVENT_CONNECT + + case BLE_GAP_EVENT_TERM_FAILURE: + { + if (client->m_conn_id != event->term_failure.conn_handle) { + return 0; + } + + log_e("Connection termination failure; rc=%d - retrying", event->term_failure.status); + if (++client->m_terminateFailCount > 2) { + ble_hs_sched_reset(BLE_HS_ECONTROLLER); + } else { + ble_gap_terminate(event->term_failure.conn_handle, BLE_ERR_REM_USER_CONN_TERM); + } + return 0; + } // BLE_GAP_EVENT_TERM_FAILURE + + case BLE_GAP_EVENT_NOTIFY_RX: + { + if (client->m_conn_id != event->notify_rx.conn_handle) { + return 0; + } + + // If a notification comes before this flag is set we might + // access a vector while it is being cleared in connect() + if (!client->m_isConnected) { + return 0; + } + + log_d("BLEClient", "Notify received for handle: %d", event->notify_rx.attr_handle); + + for (auto &myPair : client->m_servicesMap) { + // Dont waste cycles searching services without this handle in its range + if (myPair.second->getEndHandle() < event->notify_rx.attr_handle) { + continue; + } + + auto cMap = &myPair.second->m_characteristicMap; + log_d("BLEClient", "checking service %s for handle: %d", myPair.second->getUUID().toString().c_str(), event->notify_rx.attr_handle); + + auto characteristic = cMap->cbegin(); + for (; characteristic != cMap->cend(); ++characteristic) { + if (characteristic->second->m_handle == event->notify_rx.attr_handle) { + break; + } + } + + if (characteristic != cMap->cend()) { + log_d("BLEClient", "Got Notification for characteristic %s", characteristic->second->toString().c_str()); + + characteristic->second->m_semaphoreReadCharEvt.take(); + characteristic->second->m_value = String((char *)event->notify_rx.om->om_data, event->notify_rx.om->om_len); + characteristic->second->m_semaphoreReadCharEvt.give(); + + if (characteristic->second->m_notifyCallback != nullptr) { + log_d("Invoking callback for notification on characteristic %s", characteristic->second->toString().c_str()); + characteristic->second->m_notifyCallback( + characteristic->second, event->notify_rx.om->om_data, event->notify_rx.om->om_len, !event->notify_rx.indication + ); + } + break; + } + } + + return 0; + } // BLE_GAP_EVENT_NOTIFY_RX + + case BLE_GAP_EVENT_CONN_UPDATE_REQ: + case BLE_GAP_EVENT_L2CAP_UPDATE_REQ: + { + if (client->m_conn_id != event->conn_update_req.conn_handle) { + return 0; + } + log_d("Peer requesting to update connection parameters"); + log_d( + "MinInterval: %d, MaxInterval: %d, Latency: %d, Timeout: %d", event->conn_update_req.peer_params->itvl_min, + event->conn_update_req.peer_params->itvl_max, event->conn_update_req.peer_params->latency, event->conn_update_req.peer_params->supervision_timeout + ); + + rc = client->m_pClientCallbacks->onConnParamsUpdateRequest(client, event->conn_update_req.peer_params) ? 0 : BLE_ERR_CONN_PARMS; + + if (!rc && event->type == BLE_GAP_EVENT_CONN_UPDATE_REQ) { + event->conn_update_req.self_params->itvl_min = client->m_pConnParams.itvl_min; + event->conn_update_req.self_params->itvl_max = client->m_pConnParams.itvl_max; + event->conn_update_req.self_params->latency = client->m_pConnParams.latency; + event->conn_update_req.self_params->supervision_timeout = client->m_pConnParams.supervision_timeout; + } + + log_d("%s peer params", (rc == 0) ? "Accepted" : "Rejected"); + return rc; + } // BLE_GAP_EVENT_CONN_UPDATE_REQ, BLE_GAP_EVENT_L2CAP_UPDATE_REQ + + case BLE_GAP_EVENT_CONN_UPDATE: + { + if (client->m_conn_id != event->conn_update.conn_handle) { + return 0; + } + if (event->conn_update.status == 0) { + log_i("Connection parameters updated."); + } else { + log_e("Update connection parameters failed."); + } + return 0; + } // BLE_GAP_EVENT_CONN_UPDATE + + case BLE_GAP_EVENT_ENC_CHANGE: + { + if (client->m_conn_id != event->enc_change.conn_handle) { + return 0; + } + + if (event->enc_change.status == 0 || event->enc_change.status == (BLE_HS_ERR_HCI_BASE + BLE_ERR_PINKEY_MISSING)) { + struct ble_gap_conn_desc desc; + rc = ble_gap_conn_find(event->enc_change.conn_handle, &desc); + assert(rc == 0); + + if (event->enc_change.status == (BLE_HS_ERR_HCI_BASE + BLE_ERR_PINKEY_MISSING)) { + // Key is missing, try deleting. + ble_store_util_delete_peer(&desc.peer_id_addr); + } else if (BLEDevice::m_securityCallbacks != nullptr) { + BLEDevice::m_securityCallbacks->onAuthenticationComplete(&desc); + } else { + client->m_pClientCallbacks->onAuthenticationComplete(&desc); + } + } + + rc = event->enc_change.status; + break; + } //BLE_GAP_EVENT_ENC_CHANGE + + case BLE_GAP_EVENT_MTU: + { + if (client->m_conn_id != event->mtu.conn_handle) { + return 0; + } + log_i("mtu update event; conn_handle=%d mtu=%d", event->mtu.conn_handle, event->mtu.value); + rc = 0; + break; + } // BLE_GAP_EVENT_MTU + + case BLE_GAP_EVENT_PASSKEY_ACTION: + { + struct ble_sm_io pkey = {0, 0}; + + if (client->m_conn_id != event->passkey.conn_handle) { + return 0; + } + + if (event->passkey.params.action == BLE_SM_IOACT_DISP) { + pkey.action = event->passkey.params.action; + pkey.passkey = BLESecurity::m_passkey; // This is the passkey to be entered on peer + rc = ble_sm_inject_io(event->passkey.conn_handle, &pkey); + log_d("ble_sm_inject_io result: %d", rc); + + } else if (event->passkey.params.action == BLE_SM_IOACT_NUMCMP) { + log_d("Passkey on device's display: %d", event->passkey.params.numcmp); + pkey.action = event->passkey.params.action; + // Compatibility only - Do not use, should be removed the in future + if (BLEDevice::m_securityCallbacks != nullptr) { + pkey.numcmp_accept = BLEDevice::m_securityCallbacks->onConfirmPIN(event->passkey.params.numcmp); + //////////////////////////////////////////////////// + } else { + pkey.numcmp_accept = client->m_pClientCallbacks->onConfirmPIN(event->passkey.params.numcmp); + } + + rc = ble_sm_inject_io(event->passkey.conn_handle, &pkey); + log_d("ble_sm_inject_io result: %d", rc); + + //TODO: Handle out of band pairing + } else if (event->passkey.params.action == BLE_SM_IOACT_OOB) { + static uint8_t tem_oob[16] = {0}; + pkey.action = event->passkey.params.action; + for (int i = 0; i < 16; i++) { + pkey.oob[i] = tem_oob[i]; + } + rc = ble_sm_inject_io(event->passkey.conn_handle, &pkey); + log_d("ble_sm_inject_io result: %d", rc); + //////// + } else if (event->passkey.params.action == BLE_SM_IOACT_INPUT) { + log_d("Enter the passkey"); + pkey.action = event->passkey.params.action; + + // Compatibility only - Do not use, should be removed the in future + if (BLEDevice::m_securityCallbacks != nullptr) { + pkey.passkey = BLEDevice::m_securityCallbacks->onPassKeyRequest(); + ///////////////////////////////////////////// + } else { + pkey.passkey = client->m_pClientCallbacks->onPassKeyRequest(); + } + + rc = ble_sm_inject_io(event->passkey.conn_handle, &pkey); + log_d("ble_sm_inject_io result: %d", rc); + + } else if (event->passkey.params.action == BLE_SM_IOACT_NONE) { + log_d("No passkey action required"); + } + + return 0; + } // BLE_GAP_EVENT_PASSKEY_ACTION + + default: + { + return 0; + } + } // Switch + + if (pTaskData != nullptr) { + BLEUtils::taskRelease(*pTaskData, rc); + } + + return 0; +} // handleGAPEvent + /** - @brief Set the local and remote MTU size. - Should be called once after client connects if MTU size needs to be changed. - @return bool indicating if MTU was successfully set locally and on remote. -*/ -bool BLEClient::setMTU(uint16_t mtu) { - esp_err_t err = esp_ble_gatt_set_local_mtu(mtu); //First must set local MTU value. - if (err == ESP_OK) { - err = esp_ble_gattc_send_mtu_req(m_gattc_if, m_conn_id); //Once local is set successfully set remote size - if (err != ESP_OK) { - log_e("Error setting send MTU request MTU: %d err=%d", mtu, err); - return false; + * @brief Disconnect from the peer. + * @return Error code from NimBLE stack, 0 = success. + */ +int BLEClient::disconnect(uint8_t reason) { + log_d(">> disconnect()"); + int rc = 0; + + if (isConnected()) { + rc = ble_gap_terminate(m_conn_id, reason); + if (rc != 0 && rc != BLE_HS_ENOTCONN && rc != BLE_HS_EALREADY) { + m_lastErr = rc; + log_e("ble_gap_terminate failed: rc=%d %s", rc, BLEUtils::returnCodeToString(rc)); } } else { - log_e("can't set local mtu value: %d", mtu); - return false; + log_d("Not connected to any peers"); } - log_v("<< setLocalMTU"); - m_mtu = mtu; //successfully changed + log_d("<< disconnect()"); + return rc; +} // disconnect +bool BLEClientCallbacks::onConnParamsUpdateRequest(BLEClient *pClient, const ble_gap_upd_params *params) { + log_d("BLEClientCallbacks", "onConnParamsUpdateRequest: default"); return true; } -/** - * @brief Return a string representation of this client. - * @return A string representation of this client. - */ -String BLEClient::toString() { - String res = "peer address: " + m_peerAddress.toString(); - res += "\nServices:\n"; - for (auto &myPair : m_servicesMap) { - res += myPair.second->toString() + "\n"; - // myPair.second is the value - } - return res; -} // toString +uint32_t BLEClientCallbacks::onPassKeyRequest() { + log_d("onPassKeyRequest: default: 123456"); + return 123456; +} + +void BLEClientCallbacks::onAuthenticationComplete(ble_gap_conn_desc *desc) { + log_d("onAuthenticationComplete: default"); +} + +bool BLEClientCallbacks::onConfirmPIN(uint32_t pin) { + log_d("onConfirmPIN: default: true"); + return true; +} + +#endif // CONFIG_NIMBLE_ENABLED -#endif /* CONFIG_BLUEDROID_ENABLED */ +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ diff --git a/libraries/BLE/src/BLEClient.h b/libraries/BLE/src/BLEClient.h index ddb932fcd95..97ef72f4917 100644 --- a/libraries/BLE/src/BLEClient.h +++ b/libraries/BLE/src/BLEClient.h @@ -1,66 +1,128 @@ /* - * BLEDevice.h + * BLEClient.h * * Created on: Mar 22, 2017 * Author: kolban + * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on kolban's and h2zero's work) + * Description: Added support for NimBLE */ -#ifndef MAIN_BLEDEVICE_H_ -#define MAIN_BLEDEVICE_H_ +#ifndef MAIN_BLECLIENT_H_ +#define MAIN_BLECLIENT_H_ #include "soc/soc_caps.h" #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) + +/*************************************************************************** + * Common includes * + ***************************************************************************/ -#include #include #include #include -//#include "BLEExceptions.h" #include "BLERemoteService.h" #include "BLEService.h" #include "BLEAddress.h" #include "BLEAdvertisedDevice.h" +#include "BLEUtils.h" + +/*************************************************************************** + * Bluedroid includes * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) +#include +#ifndef BLE_ERR_REM_USER_CONN_TERM +#define BLE_ERR_REM_USER_CONN_TERM 0x13 +#endif +#endif + +/*************************************************************************** + * NimBLE includes and definitions * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) +#include +#include +#include +#define ESP_GATT_IF_NONE BLE_HS_CONN_HANDLE_NONE +#endif + +/*************************************************************************** + * NimBLE types * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) +typedef uint16_t esp_gatt_if_t; +#endif + +/*************************************************************************** + * Forward declarations * + ***************************************************************************/ class BLERemoteService; class BLEClientCallbacks; class BLEAdvertisedDevice; +struct BLETaskData; /** * @brief A model of a %BLE client. */ class BLEClient { public: + /*************************************************************************** + * Common public properties * + ***************************************************************************/ + + uint16_t m_appId; + + /*************************************************************************** + * Common public declarations * + ***************************************************************************/ + BLEClient(); ~BLEClient(); - bool connect(BLEAdvertisedDevice *device); bool connectTimeout(BLEAdvertisedDevice *device, uint32_t timeoutMS = portMAX_DELAY); - bool connect(BLEAddress address, esp_ble_addr_type_t type = BLE_ADDR_TYPE_PUBLIC, uint32_t timeoutMS = portMAX_DELAY); // Connect to the remote BLE Server - void disconnect(); // Disconnect from the remote BLE Server - BLEAddress getPeerAddress(); // Get the address of the remote BLE Server - int getRssi(); // Get the RSSI of the remote BLE Server - std::map *getServices(); // Get a map of the services offered by the remote BLE Server - BLERemoteService *getService(const char *uuid); // Get a reference to a specified service offered by the remote BLE server. - BLERemoteService *getService(BLEUUID uuid); // Get a reference to a specified service offered by the remote BLE server. - String getValue(BLEUUID serviceUUID, BLEUUID characteristicUUID); // Get the value of a given characteristic at a given service. - - void handleGAPEvent(esp_gap_ble_cb_event_t event, esp_ble_gap_cb_param_t *param); - - bool isConnected(); // Return true if we are connected. - + bool connect(BLEAddress address, uint8_t type = 0, uint32_t timeoutMS = portMAX_DELAY); + bool secureConnection(); + int disconnect(uint8_t reason = BLE_ERR_REM_USER_CONN_TERM); + BLEAddress getPeerAddress(); + int getRssi(); + std::map *getServices(); + BLERemoteService *getService(const char *uuid); + BLERemoteService *getService(BLEUUID uuid); + String getValue(BLEUUID serviceUUID, BLEUUID characteristicUUID); + bool isConnected(); void setClientCallbacks(BLEClientCallbacks *pClientCallbacks); - void setValue(BLEUUID serviceUUID, BLEUUID characteristicUUID, String value); // Set the value of a given characteristic at a given service. - - String toString(); // Return a string representation of this client. + void setValue(BLEUUID serviceUUID, BLEUUID characteristicUUID, String value); + String toString(); uint16_t getConnId(); esp_gatt_if_t getGattcIf(); uint16_t getMTU(); bool setMTU(uint16_t mtu); - uint16_t m_appId; + /*************************************************************************** + * Bluedroid public declarations * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) + void handleGAPEvent(esp_gap_ble_cb_event_t event, esp_ble_gap_cb_param_t *param); +#endif + + /*************************************************************************** + * NimBLE public declarations * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) + static int handleGAPEvent(struct ble_gap_event *event, void *arg); + static int serviceDiscoveredCB(uint16_t conn_handle, const struct ble_gatt_error *error, const struct ble_gatt_svc *service, void *arg); +#endif private: friend class BLEDevice; @@ -68,15 +130,14 @@ class BLEClient { friend class BLERemoteCharacteristic; friend class BLERemoteDescriptor; - void gattClientEventHandler(esp_gattc_cb_event_t event, esp_gatt_if_t gattc_if, esp_ble_gattc_cb_param_t *param); + /*************************************************************************** + * Common private properties * + ***************************************************************************/ - BLEAddress m_peerAddress = BLEAddress((uint8_t *)"\0\0\0\0\0\0"); // The BD address of the remote server. + BLEAddress m_peerAddress = BLEAddress((uint8_t *)"\0\0\0\0\0\0"); uint16_t m_conn_id; - // int m_deviceType; - esp_gatt_if_t m_gattc_if; - bool m_haveServices = false; // Have we previously obtain the set of services from the remote server. - bool m_isConnected = false; // Are we currently connected. - + bool m_haveServices = false; + bool m_isConnected = false; BLEClientCallbacks *m_pClientCallbacks; FreeRTOS::Semaphore m_semaphoreRegEvt = FreeRTOS::Semaphore("RegEvt"); FreeRTOS::Semaphore m_semaphoreOpenEvt = FreeRTOS::Semaphore("OpenEvt"); @@ -84,20 +145,76 @@ class BLEClient { FreeRTOS::Semaphore m_semaphoreRssiCmplEvt = FreeRTOS::Semaphore("RssiCmplEvt"); std::map m_servicesMap; std::map m_servicesMapByInstID; - void clearServices(); // Clear any existing services. uint16_t m_mtu = 23; -}; // class BLEDevice + + /*************************************************************************** + * Bluedroid private properties * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) + esp_gatt_if_t m_gattc_if; +#endif + + /*************************************************************************** + * NimBLE private properties * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) + int m_lastErr; + int32_t m_connectTimeout; + uint8_t m_terminateFailCount; + ble_gap_conn_params m_pConnParams; + mutable BLETaskData *m_pTaskData; +#endif + + /*************************************************************************** + * Common private declarations * + ***************************************************************************/ + + void clearServices(); + + /*************************************************************************** + * Bluedroid private declarations * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) + void gattClientEventHandler(esp_gattc_cb_event_t event, esp_gatt_if_t gattc_if, esp_ble_gattc_cb_param_t *param); +#endif + + /*************************************************************************** + * NimBLE private declarations * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) + static void dcTimerCb(ble_npl_event *event); +#endif +}; // class BLEClient /** * @brief Callbacks associated with a %BLE client. */ class BLEClientCallbacks { public: + /*************************************************************************** + * Common public declarations * + ***************************************************************************/ + virtual ~BLEClientCallbacks(){}; - virtual void onConnect(BLEClient *pClient) = 0; - virtual void onDisconnect(BLEClient *pClient) = 0; + virtual void onConnect(BLEClient *pClient); + virtual void onDisconnect(BLEClient *pClient); + + /*************************************************************************** + * NimBLE public declarations * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) + virtual bool onConnParamsUpdateRequest(BLEClient *pClient, const ble_gap_upd_params *params); + virtual uint32_t onPassKeyRequest(); + virtual void onAuthenticationComplete(ble_gap_conn_desc *desc); + virtual bool onConfirmPIN(uint32_t pin); +#endif }; -#endif /* CONFIG_BLUEDROID_ENABLED */ +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ -#endif /* MAIN_BLEDEVICE_H_ */ +#endif /* MAIN_BLECLIENT_H_ */ diff --git a/libraries/BLE/src/BLEConnInfo.h b/libraries/BLE/src/BLEConnInfo.h new file mode 100644 index 00000000000..b81ab93c28f --- /dev/null +++ b/libraries/BLE/src/BLEConnInfo.h @@ -0,0 +1,110 @@ +/* + * Copyright 2020-2025 Ryan Powell and + * esp-nimble-cpp, NimBLE-Arduino contributors. + * + * 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. + */ + +#ifndef BLECONNINFO_H_ +#define BLECONNINFO_H_ + +#if defined(CONFIG_NIMBLE_ENABLED) + +#include +#include "BLEAddress.h" + +/** + * @brief Connection information. + */ +class BLEConnInfo { +public: + /** @brief Gets the over-the-air address of the connected peer */ + BLEAddress getAddress() const { + return BLEAddress(m_desc.peer_ota_addr); + } + + /** @brief Gets the ID address of the connected peer */ + BLEAddress getIdAddress() const { + return BLEAddress(m_desc.peer_id_addr); + } + + /** @brief Gets the connection handle (also known as the connection id) of the connected peer */ + uint16_t getConnHandle() const { + return m_desc.conn_handle; + } + + /** @brief Gets the connection interval for this connection (in 1.25ms units) */ + uint16_t getConnInterval() const { + return m_desc.conn_itvl; + } + + /** @brief Gets the supervision timeout for this connection (in 10ms units) */ + uint16_t getConnTimeout() const { + return m_desc.supervision_timeout; + } + + /** @brief Gets the allowable latency for this connection (unit = number of intervals) */ + uint16_t getConnLatency() const { + return m_desc.conn_latency; + } + + /** @brief Gets the maximum transmission unit size for this connection (in bytes) */ + uint16_t getMTU() const { + return ble_att_mtu(m_desc.conn_handle); + } + + /** @brief Check if we are in the master role in this connection */ + bool isMaster() const { + return (m_desc.role == BLE_GAP_ROLE_MASTER); + } + + /** @brief Check if we are in the slave role in this connection */ + bool isSlave() const { + return (m_desc.role == BLE_GAP_ROLE_SLAVE); + } + + /** @brief Check if we are connected to a bonded peer */ + bool isBonded() const { + return (m_desc.sec_state.bonded == 1); + } + + /** @brief Check if the connection in encrypted */ + bool isEncrypted() const { + return (m_desc.sec_state.encrypted == 1); + } + + /** @brief Check if the the connection has been authenticated */ + bool isAuthenticated() const { + return (m_desc.sec_state.authenticated == 1); + } + + /** @brief Gets the key size used to encrypt the connection */ + uint8_t getSecKeySize() const { + return m_desc.sec_state.key_size; + } + +private: + friend class BLEServer; + friend class BLEClient; + friend class BLECharacteristic; + friend class BLEDescriptor; + + ble_gap_conn_desc m_desc{}; + BLEConnInfo(){}; + BLEConnInfo(ble_gap_conn_desc desc) { + m_desc = desc; + } +}; +#endif + +#endif // BLECONNINFO_H_ diff --git a/libraries/BLE/src/BLEDescriptor.cpp b/libraries/BLE/src/BLEDescriptor.cpp index 69a93e57201..51c77bb9b49 100644 --- a/libraries/BLE/src/BLEDescriptor.cpp +++ b/libraries/BLE/src/BLEDescriptor.cpp @@ -3,25 +3,50 @@ * * Created on: Jun 22, 2017 * Author: kolban + * + * Modified on: Apr 3, 2025 + * Author: lucasssvaz (based on kolban's and h2zero's work) + * Description: Added support for NimBLE */ + #include "soc/soc_caps.h" #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) + +/*************************************************************************** + * Common includes * + ***************************************************************************/ + #include #include #include #include #include "sdkconfig.h" #include +#include "BLE2904.h" #include "BLEService.h" #include "BLEDescriptor.h" #include "GeneralUtils.h" #include "esp32-hal-log.h" +/*************************************************************************** + * Common definitions * + ***************************************************************************/ + #define NULL_HANDLE (0xffff) +/*************************************************************************** + * Common global variables * + ***************************************************************************/ + +static BLEDescriptorCallbacks defaultCallbacks; + +/*************************************************************************** + * Common functions * + ***************************************************************************/ + /** * @brief BLEDescriptor constructor. */ @@ -32,13 +57,15 @@ BLEDescriptor::BLEDescriptor(const char *uuid, uint16_t len) : BLEDescriptor(BLE */ BLEDescriptor::BLEDescriptor(BLEUUID uuid, uint16_t max_len) { m_bleUUID = uuid; - m_value.attr_len = 0; // Initial length is 0. - m_value.attr_max_len = max_len; // Maximum length of the data. - m_handle = NULL_HANDLE; // Handle is initially unknown. - m_pCharacteristic = nullptr; // No initial characteristic. - m_pCallback = nullptr; // No initial callback. - + m_handle = NULL_HANDLE; // Handle is initially unknown. + m_pCharacteristic = nullptr; // No initial characteristic. + m_pCallback = nullptr; // No initial callback. + m_value.attr_len = 0; // Initial length is 0. + m_value.attr_max_len = max_len; // Maximum length of the data. m_value.attr_value = (uint8_t *)malloc(max_len); // Allocate storage for the value. +#if CONFIG_NIMBLE_ENABLED + m_removed = 0; +#endif } // BLEDescriptor /** @@ -62,6 +89,7 @@ void BLEDescriptor::executeCreate(BLECharacteristic *pCharacteristic) { m_pCharacteristic = pCharacteristic; // Save the characteristic associated with this service. +#if CONFIG_BLUEDROID_ENABLED esp_attr_control_t control; control.auto_rsp = ESP_GATT_AUTO_RSP; m_semaphoreCreateEvt.take("executeCreate"); @@ -73,6 +101,7 @@ void BLEDescriptor::executeCreate(BLECharacteristic *pCharacteristic) { } m_semaphoreCreateEvt.wait("executeCreate"); +#endif log_v("<< executeCreate"); } // executeCreate @@ -107,6 +136,118 @@ uint8_t *BLEDescriptor::getValue() { return m_value.attr_value; } // getValue +/** + * @brief Get the characteristic this descriptor belongs to. + * @return A pointer to the characteristic this descriptor belongs to. + */ +BLECharacteristic *BLEDescriptor::getCharacteristic() { + return m_pCharacteristic; +} // getCharacteristic + +/** + * @brief Set the callback handlers for this descriptor. + * @param [in] pCallbacks An instance of a callback structure used to define any callbacks for the descriptor. + */ +void BLEDescriptor::setCallbacks(BLEDescriptorCallbacks *pCallback) { + log_v(">> setCallbacks: 0x%x", (uint32_t)pCallback); + if (pCallback != nullptr) { + m_pCallback = pCallback; + } else { + m_pCallback = &defaultCallbacks; + } + log_v("<< setCallbacks"); +} // setCallbacks + +/** + * @brief Set the handle of this descriptor. + * Set the handle of this descriptor to be the supplied value. + * @param [in] handle The handle to be associated with this descriptor. + * @return N/A. + */ +void BLEDescriptor::setHandle(uint16_t handle) { +#if defined(CONFIG_BLUEDROID_ENABLED) + log_v(">> setHandle(0x%.2x): Setting descriptor handle to be 0x%.2x", handle, handle); + m_handle = handle; + log_v("<< setHandle()"); +#endif + +#if defined(CONFIG_NIMBLE_ENABLED) + log_w("NimBLE does not support manually setting the handle of a descriptor. Ignoring request."); +#endif +} // setHandle + +/** + * @brief Set the value of the descriptor. + * @param [in] data The data to set for the descriptor. + * @param [in] length The length of the data in bytes. + */ +void BLEDescriptor::setValue(uint8_t *data, size_t length) { + if (length > m_value.attr_max_len) { + log_e("Size %d too large, must be no bigger than %d", length, m_value.attr_max_len); + return; + } + + m_semaphoreSetValue.take(); + m_value.attr_len = length; + memcpy(m_value.attr_value, data, length); +#if CONFIG_BLUEDROID_ENABLED + if (m_handle != NULL_HANDLE) { + esp_ble_gatts_set_attr_value(m_handle, length, (const uint8_t *)data); + log_d("Set the value in the GATTS database using handle 0x%x", m_handle); + } +#endif + m_semaphoreSetValue.give(); +} // setValue + +/** + * @brief Set the value of the descriptor. + * @param [in] value The value of the descriptor in string form. + */ +void BLEDescriptor::setValue(String value) { + setValue((uint8_t *)value.c_str(), value.length()); +} // setValue + +void BLEDescriptor::setAccessPermissions(uint8_t perm) { + m_permissions = perm; +} + +/** + * @brief Return a string representation of the descriptor. + * @return A string representation of the descriptor. + */ +String BLEDescriptor::toString() { + char hex[5]; + snprintf(hex, sizeof(hex), "%04x", m_handle); + String res = "UUID: " + m_bleUUID.toString() + ", handle: 0x" + hex; + return res; +} // toString + +BLEDescriptorCallbacks::~BLEDescriptorCallbacks() {} + +/** + * @brief Callback function to support a read request. + * @param [in] pDescriptor The descriptor that is the source of the event. + */ +void BLEDescriptorCallbacks::onRead(BLEDescriptor *pDescriptor) { + log_d("BLEDescriptorCallbacks", ">> onRead: default"); + log_d("BLEDescriptorCallbacks", "<< onRead"); +} // onRead + +/** + * @brief Callback function to support a write request. + * @param [in] pDescriptor The descriptor that is the source of the event. + */ +void BLEDescriptorCallbacks::onWrite(BLEDescriptor *pDescriptor) { + log_d("BLEDescriptorCallbacks", ">> onWrite: default"); + log_d("BLEDescriptorCallbacks", "<< onWrite"); +} // onWrite + +/*************************************************************************** + * Bluedroid functions * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) + /** * @brief Handle GATT server events for the descripttor. * @param [in] event @@ -185,88 +326,84 @@ void BLEDescriptor::handleGATTServerEvent(esp_gatts_cb_event_t event, esp_gatt_i } // switch event } // handleGATTServerEvent -/** - * @brief Set the callback handlers for this descriptor. - * @param [in] pCallbacks An instance of a callback structure used to define any callbacks for the descriptor. - */ -void BLEDescriptor::setCallbacks(BLEDescriptorCallbacks *pCallback) { - log_v(">> setCallbacks: 0x%x", (uint32_t)pCallback); - m_pCallback = pCallback; - log_v("<< setCallbacks"); -} // setCallbacks +#endif -/** - * @brief Set the handle of this descriptor. - * Set the handle of this descriptor to be the supplied value. - * @param [in] handle The handle to be associated with this descriptor. - * @return N/A. - */ -void BLEDescriptor::setHandle(uint16_t handle) { - log_v(">> setHandle(0x%.2x): Setting descriptor handle to be 0x%.2x", handle, handle); - m_handle = handle; - log_v("<< setHandle()"); -} // setHandle +/*************************************************************************** + * NimBLE functions * + ***************************************************************************/ -/** - * @brief Set the value of the descriptor. - * @param [in] data The data to set for the descriptor. - * @param [in] length The length of the data in bytes. - */ -void BLEDescriptor::setValue(uint8_t *data, size_t length) { - if (length > ESP_GATT_MAX_ATTR_LEN) { - log_e("Size %d too large, must be no bigger than %d", length, ESP_GATT_MAX_ATTR_LEN); - return; - } - m_value.attr_len = length; - memcpy(m_value.attr_value, data, length); - if (m_handle != NULL_HANDLE) { - esp_ble_gatts_set_attr_value(m_handle, length, (const uint8_t *)data); - log_d("Set the value in the GATTS database using handle 0x%x", m_handle); - } -} // setValue +#if defined(CONFIG_NIMBLE_ENABLED) /** - * @brief Set the value of the descriptor. - * @param [in] value The value of the descriptor in string form. + * @brief Handle GATT server events for the descriptor. + * @param [in] conn_handle The connection handle. + * @param [in] attr_handle The attribute handle. + * @param [in] ctxt The GATT access context. + * @param [in] arg The argument. */ -void BLEDescriptor::setValue(String value) { - setValue((uint8_t *)value.c_str(), value.length()); -} // setValue +int BLEDescriptor::handleGATTServerEvent(uint16_t conn_handle, uint16_t attr_handle, struct ble_gatt_access_ctxt *ctxt, void *arg) { + const ble_uuid_t *uuid; + int rc; + struct ble_gap_conn_desc desc; + BLEDescriptor *pDescriptor = (BLEDescriptor *)arg; + + log_d("Descriptor %s %s event", pDescriptor->getUUID().toString().c_str(), ctxt->op == BLE_GATT_ACCESS_OP_READ_DSC ? "Read" : "Write"); + + uuid = ctxt->chr->uuid; + if (ble_uuid_cmp(uuid, &pDescriptor->getUUID().getNative()->u) == 0) { + switch (ctxt->op) { + case BLE_GATT_ACCESS_OP_READ_DSC: + { + rc = ble_gap_conn_find(conn_handle, &desc); + assert(rc == 0); + + // If the packet header is only 8 bytes this is a follow up of a long read + // so we don't want to call the onRead() callback again. + if (ctxt->om->om_pkthdr_len > 8 || pDescriptor->m_value.attr_len <= (ble_att_mtu(desc.conn_handle) - 3)) { + pDescriptor->m_pCallback->onRead(pDescriptor); + } -void BLEDescriptor::setAccessPermissions(esp_gatt_perm_t perm) { - m_permissions = perm; -} + ble_npl_hw_enter_critical(); + rc = os_mbuf_append(ctxt->om, pDescriptor->m_value.attr_value, pDescriptor->m_value.attr_len); + ble_npl_hw_exit_critical(0); + return rc == 0 ? 0 : BLE_ATT_ERR_INSUFFICIENT_RES; + } -/** - * @brief Return a string representation of the descriptor. - * @return A string representation of the descriptor. - */ -String BLEDescriptor::toString() { - char hex[5]; - snprintf(hex, sizeof(hex), "%04x", m_handle); - String res = "UUID: " + m_bleUUID.toString() + ", handle: 0x" + hex; - return res; -} // toString + case BLE_GATT_ACCESS_OP_WRITE_DSC: + { + uint16_t att_max_len = pDescriptor->m_value.attr_max_len; -BLEDescriptorCallbacks::~BLEDescriptorCallbacks() {} + if (ctxt->om->om_len > att_max_len) { + return BLE_ATT_ERR_INVALID_ATTR_VALUE_LEN; + } -/** - * @brief Callback function to support a read request. - * @param [in] pDescriptor The descriptor that is the source of the event. - */ -void BLEDescriptorCallbacks::onRead(BLEDescriptor *pDescriptor) { - log_d("BLEDescriptorCallbacks", ">> onRead: default"); - log_d("BLEDescriptorCallbacks", "<< onRead"); -} // onRead + uint8_t buf[att_max_len]; + size_t len = ctxt->om->om_len; + memcpy(buf, ctxt->om->om_data, len); + os_mbuf *next; + next = SLIST_NEXT(ctxt->om, om_next); + while (next != NULL) { + if ((len + next->om_len) > att_max_len) { + return BLE_ATT_ERR_INVALID_ATTR_VALUE_LEN; + } + memcpy(&buf[len], next->om_data, next->om_len); + len += next->om_len; + next = SLIST_NEXT(next, om_next); + } -/** - * @brief Callback function to support a write request. - * @param [in] pDescriptor The descriptor that is the source of the event. - */ -void BLEDescriptorCallbacks::onWrite(BLEDescriptor *pDescriptor) { - log_d("BLEDescriptorCallbacks", ">> onWrite: default"); - log_d("BLEDescriptorCallbacks", "<< onWrite"); -} // onWrite + pDescriptor->setValue(buf, len); + pDescriptor->m_pCallback->onWrite(pDescriptor); + return 0; + } + + default: break; + } + } + + return BLE_ATT_ERR_UNLIKELY; +} + +#endif -#endif /* CONFIG_BLUEDROID_ENABLED */ +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ diff --git a/libraries/BLE/src/BLEDescriptor.h b/libraries/BLE/src/BLEDescriptor.h index e155a1f2971..d0d34b24388 100644 --- a/libraries/BLE/src/BLEDescriptor.h +++ b/libraries/BLE/src/BLEDescriptor.h @@ -3,6 +3,10 @@ * * Created on: Jun 22, 2017 * Author: kolban + * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on kolban's and h2zero's work) + * Description: Added support for NimBLE */ #ifndef COMPONENTS_CPP_UTILS_BLEDESCRIPTOR_H_ @@ -11,13 +15,60 @@ #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) + +/*************************************************************************** + * Common includes * + ***************************************************************************/ + #include #include "BLEUUID.h" #include "BLECharacteristic.h" -#include #include "RTOS.h" +/*************************************************************************** + * Bluedroid includes * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) +#include +#endif + +/*************************************************************************** + * NimBLE includes * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) +#include +#include "BLEConnInfo.h" + +#define ESP_GATT_PERM_READ BLE_ATT_F_READ +#define ESP_GATT_PERM_WRITE BLE_ATT_F_WRITE +#define ESP_GATT_PERM_READ_ENCRYPTED BLE_ATT_F_READ_ENC +#define ESP_GATT_PERM_WRITE_ENCRYPTED BLE_ATT_F_WRITE_ENC +#define ESP_GATT_PERM_READ_AUTHORIZATION BLE_ATT_F_READ_AUTHOR +#define ESP_GATT_PERM_WRITE_AUTHORIZATION BLE_ATT_F_WRITE_AUTHOR +#define ESP_GATT_PERM_READ_ENC_MITM BLE_ATT_F_READ_AUTHEN +#define ESP_GATT_PERM_WRITE_ENC_MITM BLE_ATT_F_WRITE_AUTHEN + +#endif + +/*************************************************************************** + * NimBLE types * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) +typedef struct { + uint16_t attr_max_len; /*!< attribute max value length */ + uint16_t attr_len; /*!< attribute current value length */ + uint8_t *attr_value; /*!< the pointer to attribute value */ +} esp_attr_value_t; +#endif + +/*************************************************************************** + * Forward declarations * + ***************************************************************************/ + class BLEService; class BLECharacteristic; class BLEDescriptorCallbacks; @@ -27,33 +78,83 @@ class BLEDescriptorCallbacks; */ class BLEDescriptor { public: + /*************************************************************************** + * Common public declarations * + ***************************************************************************/ + BLEDescriptor(const char *uuid, uint16_t max_len = 100); BLEDescriptor(BLEUUID uuid, uint16_t max_len = 100); virtual ~BLEDescriptor(); - uint16_t getHandle(); // Get the handle of the descriptor. - size_t getLength(); // Get the length of the value of the descriptor. - BLEUUID getUUID(); // Get the UUID of the descriptor. - uint8_t *getValue(); // Get a pointer to the value of the descriptor. - void handleGATTServerEvent(esp_gatts_cb_event_t event, esp_gatt_if_t gatts_if, esp_ble_gatts_cb_param_t *param); + uint16_t getHandle(); // Get the handle of the descriptor. + size_t getLength(); // Get the length of the value of the descriptor. + BLEUUID getUUID(); // Get the UUID of the descriptor. + uint8_t *getValue(); // Get a pointer to the value of the descriptor. + BLECharacteristic *getCharacteristic(); // Get the characteristic that this descriptor belongs to. - void setAccessPermissions(esp_gatt_perm_t perm); // Set the permissions of the descriptor. + void setAccessPermissions(uint8_t perm); // Set the permissions of the descriptor. void setCallbacks(BLEDescriptorCallbacks *pCallbacks); // Set callbacks to be invoked for the descriptor. void setValue(uint8_t *data, size_t size); // Set the value of the descriptor as a pointer to data. void setValue(String value); // Set the value of the descriptor as a data buffer. String toString(); // Convert the descriptor to a string representation. + /*************************************************************************** + * Bluedroid public declarations * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) + void handleGATTServerEvent(esp_gatts_cb_event_t event, esp_gatt_if_t gatts_if, esp_ble_gatts_cb_param_t *param); +#endif + + /*************************************************************************** + * NimBLE public declarations * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) + static int handleGATTServerEvent(uint16_t conn_handle, uint16_t attr_handle, struct ble_gatt_access_ctxt *ctxt, void *arg); +#endif + private: friend class BLEDescriptorMap; friend class BLECharacteristic; + friend class BLEService; + friend class BLE2901; + friend class BLE2902; + friend class BLE2904; + + /*************************************************************************** + * Common private properties * + ***************************************************************************/ + BLEUUID m_bleUUID; uint16_t m_handle; + esp_attr_value_t m_value; BLEDescriptorCallbacks *m_pCallback; BLECharacteristic *m_pCharacteristic; - esp_gatt_perm_t m_permissions = ESP_GATT_PERM_READ | ESP_GATT_PERM_WRITE; + FreeRTOS::Semaphore m_semaphoreSetValue = FreeRTOS::Semaphore("SetValue"); + + /*************************************************************************** + * Bluedroid private properties * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) FreeRTOS::Semaphore m_semaphoreCreateEvt = FreeRTOS::Semaphore("CreateEvt"); - esp_attr_value_t m_value; + uint8_t m_permissions = ESP_GATT_PERM_READ | ESP_GATT_PERM_WRITE; +#endif + + /*************************************************************************** + * NimBLE private properties * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) + uint8_t m_permissions = HA_FLAG_PERM_RW; + uint8_t m_removed = 0; +#endif + + /*************************************************************************** + * Common private declarations * + ***************************************************************************/ void executeCreate(BLECharacteristic *pCharacteristic); void setHandle(uint16_t handle); @@ -68,11 +169,15 @@ class BLEDescriptor { */ class BLEDescriptorCallbacks { public: + /*************************************************************************** + * Common public declarations * + ***************************************************************************/ + virtual ~BLEDescriptorCallbacks(); virtual void onRead(BLEDescriptor *pDescriptor); virtual void onWrite(BLEDescriptor *pDescriptor); }; -#endif /* CONFIG_BLUEDROID_ENABLED */ +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ #endif /* COMPONENTS_CPP_UTILS_BLEDESCRIPTOR_H_ */ diff --git a/libraries/BLE/src/BLEDescriptorMap.cpp b/libraries/BLE/src/BLEDescriptorMap.cpp index 732fb62cdcf..e0c8b3a7cc3 100644 --- a/libraries/BLE/src/BLEDescriptorMap.cpp +++ b/libraries/BLE/src/BLEDescriptorMap.cpp @@ -3,21 +3,50 @@ * * Created on: Jun 22, 2017 * Author: kolban + * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on kolban's and h2zero's work) + * Description: Added support for NimBLE */ + #include "soc/soc_caps.h" #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) + +/*************************************************************************** + * Common includes * + ***************************************************************************/ + #include #include #include "BLECharacteristic.h" #include "BLEDescriptor.h" -#include // ESP32 BLE #ifdef ARDUINO_ARCH_ESP32 #include "esp32-hal-log.h" #endif +/*************************************************************************** + * Bluedroid includes * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) +#include // ESP32 BLE +#endif + +/*************************************************************************** + * NimBLE includes * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) +#include "host/ble_gatt.h" +#endif + +/*************************************************************************** + * Common functions * + ***************************************************************************/ + /** * @brief Return the descriptor by UUID. * @param [in] UUID The UUID to look up the descriptor. @@ -81,6 +110,24 @@ void BLEDescriptorMap::setByHandle(uint16_t handle, BLEDescriptor *pDescriptor) m_handleMap.insert(std::pair(handle, pDescriptor)); } // setByHandle +/** + * @brief Get the number of registered descriptors. + * @return The number of registered descriptors. + */ +int BLEDescriptorMap::getRegisteredDescriptorCount() { + return m_uuidMap.size(); +} + +/** + * @brief Remove a descriptor from the map. + * @param [in] pDescriptor The descriptor to remove. + * @return N/A. + */ +void BLEDescriptorMap::removeDescriptor(BLEDescriptor *pDescriptor) { + m_uuidMap.erase(pDescriptor); + m_handleMap.erase(pDescriptor->getHandle()); +} + /** * @brief Return a string representation of the descriptor map. * @return A string representation of the descriptor map. @@ -102,19 +149,6 @@ String BLEDescriptorMap::toString() { return res; } // toString -/** - * @brief Pass the GATT server event onwards to each of the descriptors found in the mapping - * @param [in] event - * @param [in] gatts_if - * @param [in] param - */ -void BLEDescriptorMap::handleGATTServerEvent(esp_gatts_cb_event_t event, esp_gatt_if_t gatts_if, esp_ble_gatts_cb_param_t *param) { - // Invoke the handler for every descriptor we have. - for (auto &myPair : m_uuidMap) { - myPair.first->handleGATTServerEvent(event, gatts_if, param); - } -} // handleGATTServerEvent - /** * @brief Get the first descriptor in the map. * @return The first descriptor in the map. @@ -142,5 +176,41 @@ BLEDescriptor *BLEDescriptorMap::getNext() { return pRet; } // getNext -#endif /* CONFIG_BLUEDROID_ENABLED */ +/*************************************************************************** + * Bluedroid functions * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) + +/** + * @brief Pass the GATT server event onwards to each of the descriptors found in the mapping + * @param [in] event + * @param [in] gatts_if + * @param [in] param + */ +void BLEDescriptorMap::handleGATTServerEvent(esp_gatts_cb_event_t event, esp_gatt_if_t gatts_if, esp_ble_gatts_cb_param_t *param) { + // Invoke the handler for every descriptor we have. + for (auto &myPair : m_uuidMap) { + myPair.first->handleGATTServerEvent(event, gatts_if, param); + } +} // handleGATTServerEvent + +#endif + +/*************************************************************************** + * NimBLE functions * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) + +void BLEDescriptorMap::handleGATTServerEvent(uint16_t conn_handle, uint16_t attr_handle, ble_gatt_access_ctxt *ctxt, void *arg) { + // Invoke the handler for every descriptor we have. + for (auto &myPair : m_uuidMap) { + myPair.first->handleGATTServerEvent(conn_handle, attr_handle, ctxt, arg); + } +} + +#endif + +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ diff --git a/libraries/BLE/src/BLEDevice.cpp b/libraries/BLE/src/BLEDevice.cpp index 186b36d6a33..014806cc32b 100644 --- a/libraries/BLE/src/BLEDevice.cpp +++ b/libraries/BLE/src/BLEDevice.cpp @@ -1,30 +1,35 @@ /* - * BLE.cpp + * BLEDevice.cpp * * Created on: Mar 16, 2017 * Author: kolban + * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on kolban's and h2zero's work) + * Description: Added support for NimBLE */ + #include "soc/soc_caps.h" #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) + +/*************************************************************************** + * Common includes * + ***************************************************************************/ + #include #include #include #include #include -#include // ESP32 BLE -#include // ESP32 BLE -#include // ESP32 BLE -#include // ESP32 BLE -#include // ESP32 BLE -#include // ESP32 BLE -#include // ESP32 BLE -#include // ESP32 ESP-IDF -#include // Part of C++ Standard library -#include // Part of C++ Standard library -#include // Part of C++ Standard library +#include + +#include +#include +#include +#include #include "BLEDevice.h" #include "BLEClient.h" @@ -37,34 +42,102 @@ #include "esp32-hal-log.h" +/*************************************************************************** + * Bluedroid includes * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) +#include +#include +#include +#include +#include +#include +#endif + +/*************************************************************************** + * NimBLE includes * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) +#ifdef CONFIG_BT_NIMBLE_LEGACY_VHCI_ENABLE +#include +#endif +#include +#include +#include +#include +#include +#include "host/ble_hs_pvcy.h" +#include "host/util/util.h" +#include "services/gap/ble_svc_gap.h" +#include "services/gatt/ble_svc_gatt.h" +#endif + +/*************************************************************************** + * Common properties * + ***************************************************************************/ + /** * Singletons for the BLEDevice. */ BLEServer *BLEDevice::m_pServer = nullptr; BLEScan *BLEDevice::m_pScan = nullptr; BLEClient *BLEDevice::m_pClient = nullptr; -bool initialized = false; -esp_ble_sec_act_t BLEDevice::m_securityLevel = (esp_ble_sec_act_t)0; +static bool initialized = false; BLESecurityCallbacks *BLEDevice::m_securityCallbacks = nullptr; uint16_t BLEDevice::m_localMTU = 23; // not sure if this variable is useful BLEAdvertising *BLEDevice::m_bleAdvertising = nullptr; uint16_t BLEDevice::m_appId = 0; std::map BLEDevice::m_connectedClientsMap; gap_event_handler BLEDevice::m_customGapHandler = nullptr; + +/*************************************************************************** + * Bluedroid properties * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) +esp_ble_sec_act_t BLEDevice::m_securityLevel = (esp_ble_sec_act_t)0; gattc_event_handler BLEDevice::m_customGattcHandler = nullptr; gatts_event_handler BLEDevice::m_customGattsHandler = nullptr; +#endif + +/*************************************************************************** + * NimBLE properties * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) +ble_gap_event_listener BLEDevice::m_listener; +BLEDeviceCallbacks BLEDevice::defaultDeviceCallbacks{}; +BLEDeviceCallbacks *BLEDevice::m_pDeviceCallbacks = &defaultDeviceCallbacks; +uint8_t BLEDevice::m_ownAddrType = BLE_OWN_ADDR_PUBLIC; +bool BLEDevice::m_synced = false; +#endif + +/*************************************************************************** + * Common functions * + ***************************************************************************/ /** * @brief Create a new instance of a client. * @return A new instance of the client. */ -/* STATIC */ BLEClient *BLEDevice::createClient() { +BLEClient *BLEDevice::createClient() { log_v(">> createClient"); -#ifndef CONFIG_GATTC_ENABLE // Check that BLE GATTC is enabled in make menuconfig - log_e("BLE GATTC is not enabled - CONFIG_GATTC_ENABLE not defined"); +#if !defined(CONFIG_GATTC_ENABLE) && !defined(CONFIG_BT_NIMBLE_ROLE_CENTRAL) + log_e("BLE Client not enabled. Check CONFIG_GATTC_ENABLE for BlueDroid or CONFIG_BT_NIMBLE_ROLE_CENTRAL for NimBLE"); abort(); #endif // CONFIG_GATTC_ENABLE - m_pClient = new BLEClient(); + +#ifdef CONFIG_NIMBLE_ENABLED + if (m_connectedClientsMap.size() >= CONFIG_BT_NIMBLE_MAX_CONNECTIONS) { + log_e("Unable to create client. Max connections reached. Cur=%d Max=%d", m_connectedClientsMap.size(), CONFIG_BT_NIMBLE_MAX_CONNECTIONS); + m_pClient = nullptr; + } else +#endif + { + m_pClient = new BLEClient(); + } log_v("<< createClient"); return m_pClient; } // createClient @@ -73,200 +146,45 @@ gatts_event_handler BLEDevice::m_customGattsHandler = nullptr; * @brief Create a new instance of a server. * @return A new instance of the server. */ -/* STATIC */ BLEServer *BLEDevice::createServer() { +BLEServer *BLEDevice::createServer() { log_v(">> createServer"); -#ifndef CONFIG_GATTS_ENABLE // Check that BLE GATTS is enabled in make menuconfig - log_e("BLE GATTS is not enabled - CONFIG_GATTS_ENABLE not defined"); +#if !defined(CONFIG_GATTS_ENABLE) && !defined(CONFIG_BT_NIMBLE_ROLE_PERIPHERAL) + log_e("BLE Server not enabled. Check CONFIG_GATTS_ENABLE for BlueDroid or CONFIG_BT_NIMBLE_ROLE_PERIPHERAL for NimBLE"); abort(); #endif // CONFIG_GATTS_ENABLE - m_pServer = new BLEServer(); - m_pServer->createApp(m_appId++); + + if (m_pServer == nullptr) { + m_pServer = new BLEServer(); + m_pServer->createApp(m_appId++); +#if defined(CONFIG_NIMBLE_ENABLED) + ble_gatts_reset(); + ble_svc_gap_init(); + ble_svc_gatt_init(); +#endif + } log_v("<< createServer"); return m_pServer; } // createServer -/** - * @brief Handle GATT server events. - * - * @param [in] event The event that has been newly received. - * @param [in] gatts_if The connection to the GATT interface. - * @param [in] param Parameters for the event. - */ -/* STATIC */ void BLEDevice::gattServerEventHandler(esp_gatts_cb_event_t event, esp_gatt_if_t gatts_if, esp_ble_gatts_cb_param_t *param) { - log_d("gattServerEventHandler [esp_gatt_if: %d] ... %s", gatts_if, BLEUtils::gattServerEventTypeToString(event).c_str()); - - BLEUtils::dumpGattServerEvent(event, gatts_if, param); - - switch (event) { - case ESP_GATTS_CONNECT_EVT: - { -#ifdef CONFIG_BLE_SMP_ENABLE // Check that BLE SMP (security) is configured in make menuconfig - if (BLEDevice::m_securityLevel) { - esp_ble_set_encryption(param->connect.remote_bda, BLEDevice::m_securityLevel); - } -#endif // CONFIG_BLE_SMP_ENABLE - break; - } // ESP_GATTS_CONNECT_EVT - - default: - { - break; - } - } // switch - - if (BLEDevice::m_pServer != nullptr) { - BLEDevice::m_pServer->handleGATTServerEvent(event, gatts_if, param); - } - - if (m_customGattsHandler != nullptr) { - m_customGattsHandler(event, gatts_if, param); - } - -} // gattServerEventHandler - -/** - * @brief Handle GATT client events. - * - * Handler for the GATT client events. - * - * @param [in] event - * @param [in] gattc_if - * @param [in] param - */ -/* STATIC */ void BLEDevice::gattClientEventHandler(esp_gattc_cb_event_t event, esp_gatt_if_t gattc_if, esp_ble_gattc_cb_param_t *param) { - - log_d("gattClientEventHandler [esp_gatt_if: %d] ... %s", gattc_if, BLEUtils::gattClientEventTypeToString(event).c_str()); - BLEUtils::dumpGattClientEvent(event, gattc_if, param); - - switch (event) { - case ESP_GATTC_CONNECT_EVT: - { -#ifdef CONFIG_BLE_SMP_ENABLE // Check that BLE SMP (security) is configured in make menuconfig - if (BLEDevice::m_securityLevel) { - esp_ble_set_encryption(param->connect.remote_bda, BLEDevice::m_securityLevel); - } -#endif // CONFIG_BLE_SMP_ENABLE - break; - } // ESP_GATTS_CONNECT_EVT - - default: break; - } // switch - for (auto &myPair : BLEDevice::getPeerDevices(true)) { - conn_status_t conn_status = (conn_status_t)myPair.second; - if (((BLEClient *)conn_status.peer_device)->getGattcIf() == gattc_if || ((BLEClient *)conn_status.peer_device)->getGattcIf() == ESP_GATT_IF_NONE - || gattc_if == ESP_GATT_IF_NONE) { - ((BLEClient *)conn_status.peer_device)->gattClientEventHandler(event, gattc_if, param); - } - } - - if (m_customGattcHandler != nullptr) { - m_customGattcHandler(event, gattc_if, param); - } - -} // gattClientEventHandler - -/** - * @brief Handle GAP events. - */ -/* STATIC */ void BLEDevice::gapEventHandler(esp_gap_ble_cb_event_t event, esp_ble_gap_cb_param_t *param) { - - BLEUtils::dumpGapEvent(event, param); - - switch (event) { - - case ESP_GAP_BLE_OOB_REQ_EVT: /* OOB request event */ log_i("ESP_GAP_BLE_OOB_REQ_EVT"); break; - case ESP_GAP_BLE_LOCAL_IR_EVT: /* BLE local IR event */ log_i("ESP_GAP_BLE_LOCAL_IR_EVT"); break; - case ESP_GAP_BLE_LOCAL_ER_EVT: /* BLE local ER event */ log_i("ESP_GAP_BLE_LOCAL_ER_EVT"); break; - case ESP_GAP_BLE_NC_REQ_EVT: /* NUMERIC CONFIRMATION */ log_i("ESP_GAP_BLE_NC_REQ_EVT"); -#ifdef CONFIG_BLE_SMP_ENABLE // Check that BLE SMP (security) is configured in make menuconfig - if (BLEDevice::m_securityCallbacks != nullptr) { - esp_ble_confirm_reply(param->ble_security.ble_req.bd_addr, BLEDevice::m_securityCallbacks->onConfirmPIN(param->ble_security.key_notif.passkey)); - } -#endif // CONFIG_BLE_SMP_ENABLE - break; - case ESP_GAP_BLE_PASSKEY_REQ_EVT: /* passkey request event */ - log_i("ESP_GAP_BLE_PASSKEY_REQ_EVT: "); - // esp_log_buffer_hex(m_remote_bda, sizeof(m_remote_bda)); -#ifdef CONFIG_BLE_SMP_ENABLE // Check that BLE SMP (security) is configured in make menuconfig - if (BLEDevice::m_securityCallbacks != nullptr) { - esp_ble_passkey_reply(param->ble_security.ble_req.bd_addr, true, BLEDevice::m_securityCallbacks->onPassKeyRequest()); - } -#endif // CONFIG_BLE_SMP_ENABLE - break; - /* - * TODO should we add white/black list comparison? - */ - case ESP_GAP_BLE_SEC_REQ_EVT: - /* send the positive(true) security response to the peer device to accept the security request. - If not accept the security request, should sent the security response with negative(false) accept value*/ - log_i("ESP_GAP_BLE_SEC_REQ_EVT"); -#ifdef CONFIG_BLE_SMP_ENABLE // Check that BLE SMP (security) is configured in make menuconfig - if (BLEDevice::m_securityCallbacks != nullptr) { - esp_ble_gap_security_rsp(param->ble_security.ble_req.bd_addr, BLEDevice::m_securityCallbacks->onSecurityRequest()); - } else { - esp_ble_gap_security_rsp(param->ble_security.ble_req.bd_addr, true); - } -#endif // CONFIG_BLE_SMP_ENABLE - break; - /* - * - */ - case ESP_GAP_BLE_PASSKEY_NOTIF_EVT: //the app will receive this evt when the IO has Output capability and the peer device IO has Input capability. - //display the passkey number to the user to input it in the peer device within 30 seconds - log_i("ESP_GAP_BLE_PASSKEY_NOTIF_EVT"); -#ifdef CONFIG_BLE_SMP_ENABLE // Check that BLE SMP (security) is configured in make menuconfig - log_i("passKey = %d", param->ble_security.key_notif.passkey); - if (BLEDevice::m_securityCallbacks != nullptr) { - BLEDevice::m_securityCallbacks->onPassKeyNotify(param->ble_security.key_notif.passkey); - } -#endif // CONFIG_BLE_SMP_ENABLE - break; - case ESP_GAP_BLE_KEY_EVT: - //shows the ble key type info share with peer device to the user. - log_d("ESP_GAP_BLE_KEY_EVT"); -#ifdef CONFIG_BLE_SMP_ENABLE // Check that BLE SMP (security) is configured in make menuconfig - log_i("key type = %s", BLESecurity::esp_key_type_to_str(param->ble_security.ble_key.key_type)); -#endif // CONFIG_BLE_SMP_ENABLE - break; - case ESP_GAP_BLE_AUTH_CMPL_EVT: log_i("ESP_GAP_BLE_AUTH_CMPL_EVT"); -#ifdef CONFIG_BLE_SMP_ENABLE // Check that BLE SMP (security) is configured in make menuconfig - if (BLEDevice::m_securityCallbacks != nullptr) { - BLEDevice::m_securityCallbacks->onAuthenticationComplete(param->ble_security.auth_cmpl); - } -#endif // CONFIG_BLE_SMP_ENABLE - break; - default: - { - break; - } - } // switch - - if (BLEDevice::m_pClient != nullptr) { - BLEDevice::m_pClient->handleGAPEvent(event, param); - } - - if (BLEDevice::m_pScan != nullptr) { - BLEDevice::getScan()->handleGAPEvent(event, param); - } - - if (m_bleAdvertising != nullptr) { - BLEDevice::getAdvertising()->handleGAPEvent(event, param); - } - - if (m_customGapHandler != nullptr) { - BLEDevice::m_customGapHandler(event, param); - } - -} // gapEventHandler - /** * @brief Get the BLE device address. * @return The BLE device address. */ -/* STATIC*/ BLEAddress BLEDevice::getAddress() { +BLEAddress BLEDevice::getAddress() { +#if defined(CONFIG_BLUEDROID_ENABLED) const uint8_t *bdAddr = esp_bt_dev_get_address(); esp_bd_addr_t addr; memcpy(addr, bdAddr, sizeof(addr)); +#endif + +#if defined(CONFIG_NIMBLE_ENABLED) + ble_addr_t addr; + int ret = ble_hs_id_copy_addr(m_ownAddrType, addr.val, NULL); + if (ret != 0) { + log_e("No BLE address found. rc=%d", ret); + return BLEAddress(); + } +#endif return BLEAddress(addr); } // getAddress @@ -275,7 +193,7 @@ gatts_event_handler BLEDevice::m_customGattsHandler = nullptr; * @return The scanning object reference. This is a singleton object. The caller should not * try and release/delete it. */ -/* STATIC */ BLEScan *BLEDevice::getScan() { +BLEScan *BLEDevice::getScan() { //log_v(">> getScan"); if (m_pScan == nullptr) { m_pScan = new BLEScan(); @@ -285,13 +203,21 @@ gatts_event_handler BLEDevice::m_customGattsHandler = nullptr; return m_pScan; } // getScan +/** + * @brief Retrieve the server object + * @return The server object + */ +BLEServer *BLEDevice::getServer() { + return m_pServer; +} // getServer + /** * @brief Get the value of a characteristic of a service on a remote device. * @param [in] bdAddress * @param [in] serviceUUID * @param [in] characteristicUUID */ -/* STATIC */ String BLEDevice::getValue(BLEAddress bdAddress, BLEUUID serviceUUID, BLEUUID characteristicUUID) { +String BLEDevice::getValue(BLEAddress bdAddress, BLEUUID serviceUUID, BLEUUID characteristicUUID) { log_v( ">> getValue: bdAddress: %s, serviceUUID: %s, characteristicUUID: %s", bdAddress.toString().c_str(), serviceUUID.toString().c_str(), characteristicUUID.toString().c_str() @@ -308,18 +234,25 @@ gatts_event_handler BLEDevice::m_customGattsHandler = nullptr; * @brief Initialize the %BLE environment. * @param deviceName The device name of the device. */ -/* STATIC */ void BLEDevice::init(String deviceName) { +void BLEDevice::init(String deviceName) { if (!initialized) { - initialized = true; // Set the initialization flag to ensure we are only initialized once. - esp_err_t errRc = ESP_OK; -#ifdef ARDUINO_ARCH_ESP32 +#if defined(CONFIG_BLUEDROID_ENABLED) +#if defined(ARDUINO_ARCH_ESP32) if (!btStart()) { errRc = ESP_FAIL; return; } #else - errRc = ::nvs_flash_init(); + btStarted(); + errRc = nvs_flash_init(); + if (errRc == ESP_ERR_NVS_NO_FREE_PAGES || errRc == ESP_ERR_NVS_NEW_VERSION_FOUND) { + errRc = nvs_flash_erase(); + if (errRc == ESP_OK) { + errRc = nvs_flash_init(); + } + } + if (errRc != ESP_OK) { log_e("nvs_flash_init: rc=%d %s", errRc, GeneralUtils::errorToString(errRc)); return; @@ -328,6 +261,7 @@ gatts_event_handler BLEDevice::m_customGattsHandler = nullptr; #ifndef CONFIG_BT_CLASSIC_ENABLED esp_bt_controller_mem_release(ESP_BT_MODE_CLASSIC_BT); #endif + esp_bt_controller_config_t bt_cfg = BT_CONTROLLER_INIT_CONFIG_DEFAULT(); errRc = esp_bt_controller_init(&bt_cfg); if (errRc != ESP_OK) { @@ -348,7 +282,7 @@ gatts_event_handler BLEDevice::m_customGattsHandler = nullptr; return; } #endif -#endif +#endif // !ARDUINO_ARCH_ESP32 esp_bluedroid_status_t bt_state = esp_bluedroid_get_status(); if (bt_state == ESP_BLUEDROID_STATUS_UNINITIALIZED) { @@ -403,6 +337,45 @@ gatts_event_handler BLEDevice::m_customGattsHandler = nullptr; return; }; #endif // CONFIG_BLE_SMP_ENABLE +#endif // CONFIG_BLUEDROID_ENABLED + +#if defined(CONFIG_NIMBLE_ENABLED) + errRc = nimble_port_init(); + if (errRc != ESP_OK) { + log_e("nimble_port_init: rc=%d %s", errRc, GeneralUtils::errorToString(errRc)); + return; + } + + // Global struct ble_hs_cfg from nimble/host/ble_hs.h needs to be initialized + ble_hs_cfg.reset_cb = BLEDevice::onReset; + ble_hs_cfg.sync_cb = BLEDevice::onSync; + ble_hs_cfg.store_status_cb = [](struct ble_store_status_event *event, void *arg) { + return m_pDeviceCallbacks->onStoreStatus(event, arg); + }; + ble_hs_cfg.sm_io_cap = BLE_HS_IO_NO_INPUT_OUTPUT; + ble_hs_cfg.sm_bonding = 0; + ble_hs_cfg.sm_mitm = 0; + ble_hs_cfg.sm_sc = 1; + ble_hs_cfg.sm_our_key_dist = BLE_SM_PAIR_KEY_DIST_ENC; + ble_hs_cfg.sm_their_key_dist = BLE_SM_PAIR_KEY_DIST_ENC; +#if MYNEWT_VAL(BLE_LL_CFG_FEAT_LL_PRIVACY) + ble_hs_cfg.sm_our_key_dist |= BLE_SM_PAIR_KEY_DIST_ID; + ble_hs_cfg.sm_their_key_dist |= BLE_SM_PAIR_KEY_DIST_ID; +#endif + + errRc = ble_svc_gap_device_name_set(deviceName.c_str()); + if (errRc != ESP_OK) { + log_e("ble_svc_gap_device_name_set: rc=%d %s", errRc, GeneralUtils::errorToString(errRc)); + } + + ble_store_config_init(); + nimble_port_freertos_init(BLEDevice::host_task); + + while (!m_synced) { + ble_npl_time_delay(1); + } +#endif // CONFIG_NIMBLE_ENABLED + initialized = true; // Set the initialization flag to ensure we are only initialized once. } vTaskDelay(200 / portTICK_PERIOD_MS); // Delay for 200 msecs as a workaround to an apparent Arduino environment issue. } // init @@ -435,7 +408,7 @@ gatts_event_handler BLEDevice::m_customGattsHandler = nullptr; * @param [in] powerType. * @param [in] powerLevel. */ -/* STATIC */ void BLEDevice::setPower(esp_power_level_t powerLevel, esp_ble_power_type_t powerType) { +void BLEDevice::setPower(esp_power_level_t powerLevel, esp_ble_power_type_t powerType) { log_v(">> setPower: %d (type: %d)", powerLevel, powerType); esp_err_t errRc = ::esp_ble_tx_power_set(powerType, powerLevel); if (errRc != ESP_OK) { @@ -444,13 +417,45 @@ gatts_event_handler BLEDevice::m_customGattsHandler = nullptr; log_v("<< setPower"); } // setPower +/** + * @brief Get the transmission power. + * @param [in] powerType The power level to set, can be one of: + * * ESP_BLE_PWR_TYPE_CONN_HDL0 = 0, For connection handle 0 + * * ESP_BLE_PWR_TYPE_CONN_HDL1 = 1, For connection handle 1 + * * ESP_BLE_PWR_TYPE_CONN_HDL2 = 2, For connection handle 2 + * * ESP_BLE_PWR_TYPE_CONN_HDL3 = 3, For connection handle 3 + * * ESP_BLE_PWR_TYPE_CONN_HDL4 = 4, For connection handle 4 + * * ESP_BLE_PWR_TYPE_CONN_HDL5 = 5, For connection handle 5 + * * ESP_BLE_PWR_TYPE_CONN_HDL6 = 6, For connection handle 6 + * * ESP_BLE_PWR_TYPE_CONN_HDL7 = 7, For connection handle 7 + * * ESP_BLE_PWR_TYPE_CONN_HDL8 = 8, For connection handle 8 + * * ESP_BLE_PWR_TYPE_ADV = 9, For advertising + * * ESP_BLE_PWR_TYPE_SCAN = 10, For scan + * * ESP_BLE_PWR_TYPE_DEFAULT = 11, For default, if not set other, it will use default value + * @return the power level currently used by the type specified. + */ + +int BLEDevice::getPower(esp_ble_power_type_t powerType) { + switch (esp_ble_tx_power_get(powerType)) { + case ESP_PWR_LVL_N12: return -12; + case ESP_PWR_LVL_N9: return -9; + case ESP_PWR_LVL_N6: return -6; + case ESP_PWR_LVL_N3: return -6; + case ESP_PWR_LVL_N0: return 0; + case ESP_PWR_LVL_P3: return 3; + case ESP_PWR_LVL_P6: return 6; + case ESP_PWR_LVL_P9: return 9; + default: return -128; + } +} // getPower + /** * @brief Set the value of a characteristic of a service on a remote device. * @param [in] bdAddress * @param [in] serviceUUID * @param [in] characteristicUUID */ -/* STATIC */ void BLEDevice::setValue(BLEAddress bdAddress, BLEUUID serviceUUID, BLEUUID characteristicUUID, String value) { +void BLEDevice::setValue(BLEAddress bdAddress, BLEUUID serviceUUID, BLEUUID characteristicUUID, String value) { log_v( ">> setValue: bdAddress: %s, serviceUUID: %s, characteristicUUID: %s", bdAddress.toString().c_str(), serviceUUID.toString().c_str(), characteristicUUID.toString().c_str() @@ -465,7 +470,7 @@ gatts_event_handler BLEDevice::m_customGattsHandler = nullptr; * @brief Return a string representation of the nature of this device. * @return A string representation of the nature of this device. */ -/* STATIC */ String BLEDevice::toString() { +String BLEDevice::toString() { String res = "BD Address: " + getAddress().toString(); return res; } // toString @@ -476,14 +481,27 @@ gatts_event_handler BLEDevice::m_customGattsHandler = nullptr; */ void BLEDevice::whiteListAdd(BLEAddress address) { log_v(">> whiteListAdd: %s", address.toString().c_str()); +#ifdef CONFIG_BLUEDROID_ENABLED #ifdef ESP_IDF_VERSION_MAJOR - esp_err_t errRc = esp_ble_gap_update_whitelist(true, *address.getNative(), BLE_WL_ADDR_TYPE_PUBLIC); // HACK!!! True to add an entry. + esp_err_t errRc = esp_ble_gap_update_whitelist(true, address.getNative(), BLE_WL_ADDR_TYPE_PUBLIC); // HACK!!! True to add an entry. #else - esp_err_t errRc = esp_ble_gap_update_whitelist(true, *address.getNative()); // True to add an entry. + esp_err_t errRc = esp_ble_gap_update_whitelist(true, address.getNative()); // True to add an entry. #endif if (errRc != ESP_OK) { log_e("esp_ble_gap_update_whitelist: rc=%d %s", errRc, GeneralUtils::errorToString(errRc)); } +#endif + +#if defined(CONFIG_NIMBLE_ENABLED) + if (!BLEDevice::onWhiteList(address)) { + m_whiteList.push_back(address); + int errRc = ble_gap_wl_set(reinterpret_cast(&m_whiteList[0]), m_whiteList.size()); + if (errRc != 0) { + log_e("Failed adding to whitelist rc=%d", errRc); + m_whiteList.pop_back(); + } + } +#endif log_v("<< whiteListAdd"); } // whiteListAdd @@ -493,28 +511,36 @@ void BLEDevice::whiteListAdd(BLEAddress address) { */ void BLEDevice::whiteListRemove(BLEAddress address) { log_v(">> whiteListRemove: %s", address.toString().c_str()); +#ifdef CONFIG_BLUEDROID_ENABLED #ifdef ESP_IDF_VERSION_MAJOR - esp_err_t errRc = esp_ble_gap_update_whitelist(false, *address.getNative(), BLE_WL_ADDR_TYPE_PUBLIC); // HACK!!! False to remove an entry. + esp_err_t errRc = esp_ble_gap_update_whitelist(false, address.getNative(), BLE_WL_ADDR_TYPE_PUBLIC); // HACK!!! False to remove an entry. #else - esp_err_t errRc = esp_ble_gap_update_whitelist(false, *address.getNative()); // False to remove an entry. + esp_err_t errRc = esp_ble_gap_update_whitelist(false, address.getNative()); // False to remove an entry. #endif if (errRc != ESP_OK) { log_e("esp_ble_gap_update_whitelist: rc=%d %s", errRc, GeneralUtils::errorToString(errRc)); } +#endif + +#if defined(CONFIG_NIMBLE_ENABLED) + for (auto it = m_whiteList.begin(); it < m_whiteList.end(); ++it) { + if (*it == address) { + m_whiteList.erase(it); + int errRc = ble_gap_wl_set(reinterpret_cast(&m_whiteList[0]), m_whiteList.size()); + if (errRc != 0) { + m_whiteList.push_back(address); + log_e("Failed removing from whitelist rc=%d", errRc); + } + std::vector(m_whiteList).swap(m_whiteList); + } + } +#endif log_v("<< whiteListRemove"); } // whiteListRemove /* - * @brief Set encryption level that will be negotiated with peer device durng connection - * @param [in] level Requested encryption level - */ -void BLEDevice::setEncryptionLevel(esp_ble_sec_act_t level) { - BLEDevice::m_securityLevel = level; -} - -/* - * @brief Set callbacks that will be used to handle encryption negotiation events and authentication events - * @param [in] cllbacks Pointer to BLESecurityCallbacks class callback + * @brief Set callbacks that will be used to handle encryption negotiation events and authentication events + * @param [in] callbacks Pointer to BLESecurityCallbacks class callback */ void BLEDevice::setSecurityCallbacks(BLESecurityCallbacks *callbacks) { BLEDevice::m_securityCallbacks = callbacks; @@ -526,11 +552,19 @@ void BLEDevice::setSecurityCallbacks(BLESecurityCallbacks *callbacks) { */ esp_err_t BLEDevice::setMTU(uint16_t mtu) { log_v(">> setLocalMTU: %d", mtu); + +#ifdef CONFIG_BLUEDROID_ENABLED esp_err_t err = esp_ble_gatt_set_local_mtu(mtu); +#endif + +#if defined(CONFIG_NIMBLE_ENABLED) + int err = ble_att_set_preferred_mtu(mtu); +#endif + if (err == ESP_OK) { m_localMTU = mtu; } else { - log_e("can't set local mtu value: %d", mtu); + log_e("can't set local mtu value: %d, rc=%d", mtu, err); } log_v("<< setLocalMTU"); return err; @@ -574,10 +608,23 @@ std::map BLEDevice::getPeerDevices(bool _client) { return m_connectedClientsMap; } +BLEClient *BLEDevice::getClientByID(uint16_t conn_id) { + return BLEDevice::getClientByGattIf(conn_id); +} + BLEClient *BLEDevice::getClientByGattIf(uint16_t conn_id) { return (BLEClient *)m_connectedClientsMap.find(conn_id)->second.peer_device; } +BLEClient *BLEDevice::getClientByAddress(BLEAddress address) { + for (auto &it : m_connectedClientsMap) { + if (((BLEClient *)it.second.peer_device)->getPeerAddress() == address) { + return (BLEClient *)it.second.peer_device; + } + } + return nullptr; +} + void BLEDevice::updatePeerDevice(void *peer, bool _client, uint16_t conn_id) { log_d("update conn_id: %d, GATT role: %s", conn_id, _client ? "client" : "server"); std::map::iterator it = m_connectedClientsMap.find(ESP_GATT_IF_NONE); @@ -619,20 +666,32 @@ void BLEDevice::removePeerDevice(uint16_t conn_id, bool _client) { * @brief de-Initialize the %BLE environment. * @param release_memory release the internal BT stack memory */ -/* STATIC */ void BLEDevice::deinit(bool release_memory) { +void BLEDevice::deinit(bool release_memory) { if (!initialized) { return; } +#ifdef CONFIG_BLUEDROID_ENABLED esp_bluedroid_disable(); esp_bluedroid_deinit(); +#endif + +#if defined(CONFIG_NIMBLE_ENABLED) + nimble_port_stop(); + nimble_port_deinit(); +#endif + esp_bt_controller_disable(); esp_bt_controller_deinit(); + #ifdef ARDUINO_ARCH_ESP32 if (release_memory) { - esp_bt_controller_mem_release(ESP_BT_MODE_BTDM - ); // <-- require tests because we released classic BT memory and this can cause crash (most likely not, esp-idf takes care of it) + // Require tests because we released classic BT memory and this can cause crash (most likely not, esp-idf takes care of it) + esp_bt_controller_mem_release(ESP_BT_MODE_BTDM); } else { +#ifdef CONFIG_NIMBLE_ENABLED + m_synced = false; +#endif initialized = false; } #endif @@ -640,8 +699,195 @@ void BLEDevice::removePeerDevice(uint16_t conn_id, bool _client) { void BLEDevice::setCustomGapHandler(gap_event_handler handler) { m_customGapHandler = handler; +#ifdef CONFIG_NIMBLE_ENABLED + int rc = ble_gap_event_listener_register(&m_listener, handler, NULL); + if (rc == BLE_HS_EALREADY) { + log_i("Already listening to GAP events."); + } else if (rc != 0) { + log_e("ble_gap_event_listener_register: rc=%d %s", rc, GeneralUtils::errorToString(rc)); + } +#endif } +/*************************************************************************** + * Bluedroid functions * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) + +/** + * @brief Handle GATT server events. + * + * @param [in] event The event that has been newly received. + * @param [in] gatts_if The connection to the GATT interface. + * @param [in] param Parameters for the event. + */ +void BLEDevice::gattServerEventHandler(esp_gatts_cb_event_t event, esp_gatt_if_t gatts_if, esp_ble_gatts_cb_param_t *param) { + log_d("gattServerEventHandler [esp_gatt_if: %d] ... %s", gatts_if, BLEUtils::gattServerEventTypeToString(event).c_str()); + + BLEUtils::dumpGattServerEvent(event, gatts_if, param); + + switch (event) { + case ESP_GATTS_CONNECT_EVT: + { +#ifdef CONFIG_BLE_SMP_ENABLE // Check that BLE SMP (security) is configured in make menuconfig + if (BLEDevice::m_securityLevel) { + esp_ble_set_encryption(param->connect.remote_bda, BLEDevice::m_securityLevel); + } +#endif // CONFIG_BLE_SMP_ENABLE + break; + } // ESP_GATTS_CONNECT_EVT + + default: + { + break; + } + } // switch + + if (BLEDevice::m_pServer != nullptr) { + BLEDevice::m_pServer->handleGATTServerEvent(event, gatts_if, param); + } + + if (m_customGattsHandler != nullptr) { + m_customGattsHandler(event, gatts_if, param); + } + +} // gattServerEventHandler + +/** + * @brief Handle GATT client events. + * + * Handler for the GATT client events. + * + * @param [in] event + * @param [in] gattc_if + * @param [in] param + */ +void BLEDevice::gattClientEventHandler(esp_gattc_cb_event_t event, esp_gatt_if_t gattc_if, esp_ble_gattc_cb_param_t *param) { + + log_d("gattClientEventHandler [esp_gatt_if: %d] ... %s", gattc_if, BLEUtils::gattClientEventTypeToString(event).c_str()); + BLEUtils::dumpGattClientEvent(event, gattc_if, param); + + switch (event) { + case ESP_GATTC_CONNECT_EVT: + { +#ifdef CONFIG_BLE_SMP_ENABLE // Check that BLE SMP (security) is configured in make menuconfig + if (BLEDevice::m_securityLevel) { + esp_ble_set_encryption(param->connect.remote_bda, BLEDevice::m_securityLevel); + } +#endif // CONFIG_BLE_SMP_ENABLE + break; + } // ESP_GATTS_CONNECT_EVT + + default: break; + } // switch + for (auto &myPair : BLEDevice::getPeerDevices(true)) { + conn_status_t conn_status = (conn_status_t)myPair.second; + if (((BLEClient *)conn_status.peer_device)->getGattcIf() == gattc_if || ((BLEClient *)conn_status.peer_device)->getGattcIf() == ESP_GATT_IF_NONE + || gattc_if == ESP_GATT_IF_NONE) { + ((BLEClient *)conn_status.peer_device)->gattClientEventHandler(event, gattc_if, param); + } + } + + if (m_customGattcHandler != nullptr) { + m_customGattcHandler(event, gattc_if, param); + } + +} // gattClientEventHandler + +/** + * @brief Handle GAP events. + */ +void BLEDevice::gapEventHandler(esp_gap_ble_cb_event_t event, esp_ble_gap_cb_param_t *param) { + + BLEUtils::dumpGapEvent(event, param); + + switch (event) { + + case ESP_GAP_BLE_OOB_REQ_EVT: /* OOB request event */ log_i("ESP_GAP_BLE_OOB_REQ_EVT"); break; + case ESP_GAP_BLE_LOCAL_IR_EVT: /* BLE local IR event */ log_i("ESP_GAP_BLE_LOCAL_IR_EVT"); break; + case ESP_GAP_BLE_LOCAL_ER_EVT: /* BLE local ER event */ log_i("ESP_GAP_BLE_LOCAL_ER_EVT"); break; + case ESP_GAP_BLE_NC_REQ_EVT: /* NUMERIC CONFIRMATION */ log_i("ESP_GAP_BLE_NC_REQ_EVT"); +#ifdef CONFIG_BLE_SMP_ENABLE // Check that BLE SMP (security) is configured in make menuconfig + if (BLEDevice::m_securityCallbacks != nullptr) { + esp_ble_confirm_reply(param->ble_security.ble_req.bd_addr, BLEDevice::m_securityCallbacks->onConfirmPIN(param->ble_security.key_notif.passkey)); + } +#endif // CONFIG_BLE_SMP_ENABLE + break; + case ESP_GAP_BLE_PASSKEY_REQ_EVT: /* passkey request event */ + log_i("ESP_GAP_BLE_PASSKEY_REQ_EVT: "); + // esp_log_buffer_hex(m_remote_bda, sizeof(m_remote_bda)); +#ifdef CONFIG_BLE_SMP_ENABLE // Check that BLE SMP (security) is configured in make menuconfig + if (BLEDevice::m_securityCallbacks != nullptr) { + esp_ble_passkey_reply(param->ble_security.ble_req.bd_addr, true, BLEDevice::m_securityCallbacks->onPassKeyRequest()); + } +#endif // CONFIG_BLE_SMP_ENABLE + break; + /* + * TODO should we add white/black list comparison? + */ + case ESP_GAP_BLE_SEC_REQ_EVT: + /* send the positive(true) security response to the peer device to accept the security request. + If not accept the security request, should sent the security response with negative(false) accept value*/ + log_i("ESP_GAP_BLE_SEC_REQ_EVT"); +#ifdef CONFIG_BLE_SMP_ENABLE // Check that BLE SMP (security) is configured in make menuconfig + if (BLEDevice::m_securityCallbacks != nullptr) { + esp_ble_gap_security_rsp(param->ble_security.ble_req.bd_addr, BLEDevice::m_securityCallbacks->onSecurityRequest()); + } else { + esp_ble_gap_security_rsp(param->ble_security.ble_req.bd_addr, true); + } +#endif // CONFIG_BLE_SMP_ENABLE + break; + /* + * + */ + case ESP_GAP_BLE_PASSKEY_NOTIF_EVT: //the app will receive this evt when the IO has Output capability and the peer device IO has Input capability. + //display the passkey number to the user to input it in the peer device within 30 seconds + log_i("ESP_GAP_BLE_PASSKEY_NOTIF_EVT"); +#ifdef CONFIG_BLE_SMP_ENABLE // Check that BLE SMP (security) is configured in make menuconfig + log_i("passKey = %d", param->ble_security.key_notif.passkey); + if (BLEDevice::m_securityCallbacks != nullptr) { + BLEDevice::m_securityCallbacks->onPassKeyNotify(param->ble_security.key_notif.passkey); + } +#endif // CONFIG_BLE_SMP_ENABLE + break; + case ESP_GAP_BLE_KEY_EVT: + //shows the ble key type info share with peer device to the user. + log_d("ESP_GAP_BLE_KEY_EVT"); +#ifdef CONFIG_BLE_SMP_ENABLE // Check that BLE SMP (security) is configured in make menuconfig + log_i("key type = %s", BLESecurity::esp_key_type_to_str(param->ble_security.ble_key.key_type)); +#endif // CONFIG_BLE_SMP_ENABLE + break; + case ESP_GAP_BLE_AUTH_CMPL_EVT: log_i("ESP_GAP_BLE_AUTH_CMPL_EVT"); +#ifdef CONFIG_BLE_SMP_ENABLE // Check that BLE SMP (security) is configured in make menuconfig + if (BLEDevice::m_securityCallbacks != nullptr) { + BLEDevice::m_securityCallbacks->onAuthenticationComplete(param->ble_security.auth_cmpl); + } +#endif // CONFIG_BLE_SMP_ENABLE + break; + default: + { + break; + } + } // switch + + if (BLEDevice::m_pClient != nullptr) { + BLEDevice::m_pClient->handleGAPEvent(event, param); + } + + if (BLEDevice::m_pScan != nullptr) { + BLEDevice::getScan()->handleGAPEvent(event, param); + } + + if (m_bleAdvertising != nullptr) { + BLEDevice::getAdvertising()->handleGAPEvent(event, param); + } + + if (m_customGapHandler != nullptr) { + BLEDevice::m_customGapHandler(event, param); + } + +} // gapEventHandler void BLEDevice::setCustomGattcHandler(gattc_event_handler handler) { m_customGattcHandler = handler; } @@ -650,5 +896,170 @@ void BLEDevice::setCustomGattsHandler(gatts_event_handler handler) { m_customGattsHandler = handler; } -#endif /* CONFIG_BLUEDROID_ENABLED */ +/* + * @brief Set encryption level that will be negotiated with peer device durng connection + * @param [in] level Requested encryption level + */ +void BLEDevice::setEncryptionLevel(esp_ble_sec_act_t level) { + BLEDevice::m_securityLevel = level; +} + +#endif + +/*************************************************************************** + * NimBLE functions * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) + +/** + * @brief Checks if a peer device is whitelisted. + * @param [in] address The address to check for in the whitelist. + * @returns True if the address is in the whitelist. + */ +bool BLEDevice::onWhiteList(BLEAddress &address) { + for (auto &addr : m_whiteList) { + if (addr == address) { + return true; + } + } + return false; +} + +void BLEDevice::host_task(void *param) { + log_i("NimBLE host task started"); + nimble_port_run(); // This function will return only when nimble_port_stop() is executed + nimble_port_freertos_deinit(); +} + +void BLEDevice::onReset(int reason) { + if (!m_synced) { + return; + } + + m_synced = false; + + log_i("onReset, reason=%d, %s", reason, BLEUtils::returnCodeToString(reason)); + + if (initialized) { + if (m_pScan != nullptr) { + m_pScan->onHostReset(); + } + } +} + +void BLEDevice::onSync() { + log_d("onSync"); + + if (m_synced) { + log_d("onSync: already synced"); + return; + } + + int rc = ble_hs_util_ensure_addr(0); + if (rc == 0) { + rc = ble_hs_util_ensure_addr(1); + } + + if (rc != 0) { + log_e("onSync: failed to ensure BLE address. rc=%d", rc); + return; + } + + rc = ble_hs_id_copy_addr(BLE_OWN_ADDR_PUBLIC, NULL, NULL); + if (rc != 0) { + log_d("onSync: no public address available"); + m_ownAddrType = BLE_OWN_ADDR_RANDOM; + } + + // Yield for housekeeping tasks before returning to operations. + // Occasionally triggers exception without. + ble_npl_time_delay(1); + + m_synced = true; + + if (initialized) { + if (m_pScan != nullptr) { + m_pScan->onHostSync(); + } + if (m_bleAdvertising != nullptr) { + m_bleAdvertising->onHostSync(); + } + } +} + +void BLEDevice::setDeviceCallbacks(BLEDeviceCallbacks *cb) { + if (cb == nullptr) { + m_pDeviceCallbacks = &defaultDeviceCallbacks; + } else { + m_pDeviceCallbacks = cb; + } +} + +/** + * @brief Sets the address type to use. + * @param [in] type Bluetooth Device address type. + * The available types are defined as: + * * 0x00: BLE_OWN_ADDR_PUBLIC - Public address; Uses the hardware static address. + * * 0x01: BLE_OWN_ADDR_RANDOM - Random static address; Uses the hardware or generated random static address. + * * 0x02: BLE_OWN_ADDR_RPA_PUBLIC_DEFAULT - Resolvable private address, defaults to public if no RPA available. + * * 0x03: BLE_OWN_ADDR_RPA_RANDOM_DEFAULT - Resolvable private address, defaults to random static if no RPA available. + */ +bool BLEDevice::setOwnAddrType(uint8_t type) { + int rc = ble_hs_id_copy_addr(type & 1, NULL, NULL); // Odd values are random + if (rc != 0) { + log_e("Unable to set address type %d, rc=%d", type, rc); + return false; + } + + m_ownAddrType = type; + + if (type == BLE_OWN_ADDR_RPA_PUBLIC_DEFAULT || type == BLE_OWN_ADDR_RPA_RANDOM_DEFAULT) { +#ifdef CONFIG_IDF_TARGET_ESP32 + // esp32 controller does not support RPA so we must use the random static for calls to the stack + // the host will take care of the random private address generation/setting. + m_ownAddrType = BLE_OWN_ADDR_RANDOM; + rc = ble_hs_pvcy_rpa_config(NIMBLE_HOST_ENABLE_RPA); +#endif + } else { +#ifdef CONFIG_IDF_TARGET_ESP32 + rc = ble_hs_pvcy_rpa_config(NIMBLE_HOST_DISABLE_PRIVACY); +#endif + } + return rc == 0; +} // setOwnAddrType + +/** + * @brief Set the device address to use. + * @param [in] addr The address to set. + * @return True if the address was set successfully. + * @details To use the address generated the address type must be set to random with `setOwnAddrType`. + */ +bool BLEDevice::setOwnAddr(BLEAddress &addr) { + return setOwnAddr(addr.getNative()); +} // setOwnAddr + +/** + * @brief Set the device address to use. + * @param [in] addr The address to set. + * @return True if the address was set successfully. + * @details To use the address generated the address type must be set to random with `setOwnAddrType`. + */ +bool BLEDevice::setOwnAddr(uint8_t *addr) { + int rc = ble_hs_id_set_rnd(addr); + if (rc != 0) { + log_e("Failed to set address, rc=%d", rc); + return false; + } + return true; +} // setOwnAddr + +int BLEDeviceCallbacks::onStoreStatus(struct ble_store_status_event *event, void *arg) { + log_d("onStoreStatus: default"); + return ble_store_util_status_rr(event, arg); +} + +#endif + +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ diff --git a/libraries/BLE/src/BLEDevice.h b/libraries/BLE/src/BLEDevice.h index 01bf143c101..66cfa2b371a 100644 --- a/libraries/BLE/src/BLEDevice.h +++ b/libraries/BLE/src/BLEDevice.h @@ -3,6 +3,10 @@ * * Created on: Mar 16, 2017 * Author: kolban + * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on kolban's and h2zero's work) + * Description: Added support for NimBLE */ #ifndef MAIN_BLEDevice_H_ @@ -11,88 +15,252 @@ #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) -#include // ESP32 BLE -#include // ESP32 BLE -#include // Part of C++ STL +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) + +/*************************************************************************** + * Common includes * + ***************************************************************************/ + +#include #include #include - #include "BLEServer.h" #include "BLEClient.h" #include "BLEUtils.h" #include "BLEScan.h" +#include "BLEAdvertising.h" +#include "BLESecurity.h" #include "BLEAddress.h" +#include "BLEUtils.h" -/** - * @brief BLE functions. - */ +/*************************************************************************** + * Bluedroid includes * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) +#include +#include +#include +#endif + +/*************************************************************************** + * NimBLE includes and definitions * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) +#include +#define ESP_GATT_IF_NONE BLE_HS_CONN_HANDLE_NONE + +// NimBLE configuration compatibility macros +#if defined(CONFIG_SCAN_DUPLICATE_BY_DEVICE_ADDR) && !defined(CONFIG_BTDM_SCAN_DUPL_TYPE_DEVICE) +#define CONFIG_BTDM_SCAN_DUPL_TYPE_DEVICE CONFIG_SCAN_DUPLICATE_BY_DEVICE_ADDR +#endif + +#if defined(CONFIG_SCAN_DUPLICATE_BY_ADV_DATA) && !defined(CONFIG_BTDM_SCAN_DUPL_TYPE_DATA) +#define CONFIG_BTDM_SCAN_DUPL_TYPE_DATA CONFIG_SCAN_DUPLICATE_BY_ADV_DATA +#endif + +#if defined(CONFIG_SCAN_DUPLICATE_BY_ADV_DATA_AND_DEVICE_ADDR) && !defined(CONFIG_BTDM_SCAN_DUPL_TYPE_DATA_DEVICE) +#define CONFIG_BTDM_SCAN_DUPL_TYPE_DATA_DEVICE CONFIG_SCAN_DUPLICATE_BY_ADV_DATA_AND_DEVICE_ADDR +#endif + +#if defined(CONFIG_SCAN_DUPLICATE_TYPE) && !defined(CONFIG_BTDM_SCAN_DUPL_TYPE) +#define CONFIG_BTDM_SCAN_DUPL_TYPE CONFIG_SCAN_DUPLICATE_TYPE +#endif + +#if defined(CONFIG_BT_CTRL_SCAN_DUPL_TYPE) && !defined(CONFIG_BTDM_SCAN_DUPL_TYPE) +#define CONFIG_BTDM_SCAN_DUPL_TYPE CONFIG_BT_CTRL_SCAN_DUPL_TYPE +#endif + +#if defined(CONFIG_BT_LE_SCAN_DUPL_TYPE) && !defined(CONFIG_BTDM_SCAN_DUPL_TYPE) +#define CONFIG_BTDM_SCAN_DUPL_TYPE CONFIG_BT_LE_SCAN_DUPL_TYPE +#endif + +#if defined(CONFIG_DUPLICATE_SCAN_CACHE_SIZE) && !defined(CONFIG_BTDM_SCAN_DUPL_CACHE_SIZE) +#define CONFIG_BTDM_SCAN_DUPL_CACHE_SIZE CONFIG_DUPLICATE_SCAN_CACHE_SIZE +#endif + +#if defined(CONFIG_BT_CTRL_SCAN_DUPL_CACHE_SIZE) && !defined(CONFIG_BTDM_SCAN_DUPL_CACHE_SIZE) +#define CONFIG_BTDM_SCAN_DUPL_CACHE_SIZE CONFIG_BT_CTRL_SCAN_DUPL_CACHE_SIZE +#endif + +#if defined(CONFIG_BT_LE_LL_DUP_SCAN_LIST_COUNT) && !defined(CONFIG_BTDM_SCAN_DUPL_CACHE_SIZE) +#define CONFIG_BTDM_SCAN_DUPL_CACHE_SIZE CONFIG_BT_LE_LL_DUP_SCAN_LIST_COUNT +#endif + +#if defined(CONFIG_NIMBLE_MAX_CONNECTIONS) && !defined(CONFIG_BT_NIMBLE_MAX_CONNECTIONS) +#define CONFIG_BT_NIMBLE_MAX_CONNECTIONS CONFIG_NIMBLE_MAX_CONNECTIONS +#endif + +#endif + +/*************************************************************************** + * Forward declarations * + ***************************************************************************/ + +class BLEAddress; +class BLEDeviceCallbacks; +class BLESecurityCallbacks; +class BLEServer; +class BLEScan; +class BLEAdvertising; +class BLEClient; +class BLESecurity; + +/*************************************************************************** + * Bluedroid type definitions * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) typedef void (*gap_event_handler)(esp_gap_ble_cb_event_t event, esp_ble_gap_cb_param_t *param); typedef void (*gattc_event_handler)(esp_gattc_cb_event_t event, esp_gatt_if_t gattc_if, esp_ble_gattc_cb_param_t *param); typedef void (*gatts_event_handler)(esp_gatts_cb_event_t event, esp_gatt_if_t gattc_if, esp_ble_gatts_cb_param_t *param); +#endif + +/*************************************************************************** + * NimBLE type definitions and externals * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) +extern "C" void ble_store_config_init(void); +typedef int (*gap_event_handler)(struct ble_gap_event *event, void *param); +#endif class BLEDevice { public: - static BLEClient *createClient(); // Create a new BLE client. - static BLEServer *createServer(); // Create a new BLE server. - static BLEAddress getAddress(); // Retrieve our own local BD address. - static BLEScan *getScan(); // Get the scan object - static String getValue(BLEAddress bdAddress, BLEUUID serviceUUID, BLEUUID characteristicUUID); // Get the value of a characteristic of a service on a server. - static void init(String deviceName); // Initialize the local BLE environment. - static void setPower(esp_power_level_t powerLevel, esp_ble_power_type_t powerType = ESP_BLE_PWR_TYPE_DEFAULT); // Set our power level. - static void setValue( - BLEAddress bdAddress, BLEUUID serviceUUID, BLEUUID characteristicUUID, String value - ); // Set the value of a characteristic on a service on a server. - static String toString(); // Return a string representation of our device. - static void whiteListAdd(BLEAddress address); // Add an entry to the BLE white list. - static void whiteListRemove(BLEAddress address); // Remove an entry from the BLE white list. - static void setEncryptionLevel(esp_ble_sec_act_t level); + /*************************************************************************** + * Common public properties * + ***************************************************************************/ + + static uint16_t m_appId; + static uint16_t m_localMTU; + static gap_event_handler m_customGapHandler; + + /*************************************************************************** + * Bluedroid public properties * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) + static esp_ble_sec_act_t m_securityLevel; + static gattc_event_handler m_customGattcHandler; + static gatts_event_handler m_customGattsHandler; +#endif + + /*************************************************************************** + * Common public declarations * + ***************************************************************************/ + + static BLEClient *createClient(); + static BLEServer *createServer(); + static BLEAddress getAddress(); + static BLEServer *getServer(); + static BLEScan *getScan(); + static String getValue(BLEAddress bdAddress, BLEUUID serviceUUID, BLEUUID characteristicUUID); + static void init(String deviceName); + static void setPower(esp_power_level_t powerLevel, esp_ble_power_type_t powerType = ESP_BLE_PWR_TYPE_DEFAULT); + static int getPower(esp_ble_power_type_t powerType = ESP_BLE_PWR_TYPE_DEFAULT); + static void setValue(BLEAddress bdAddress, BLEUUID serviceUUID, BLEUUID characteristicUUID, String value); + static String toString(); + static void whiteListAdd(BLEAddress address); + static void whiteListRemove(BLEAddress address); static void setSecurityCallbacks(BLESecurityCallbacks *pCallbacks); static esp_err_t setMTU(uint16_t mtu); static uint16_t getMTU(); - static bool getInitialized(); // Returns the state of the device, is it initialized or not? - /* move advertising to BLEDevice for saving ram and flash in beacons */ + static bool getInitialized(); static BLEAdvertising *getAdvertising(); static void startAdvertising(); static void stopAdvertising(); - static uint16_t m_appId; - /* multi connect */ static std::map getPeerDevices(bool client); static void addPeerDevice(void *peer, bool is_client, uint16_t conn_id); static void updatePeerDevice(void *peer, bool _client, uint16_t conn_id); static void removePeerDevice(uint16_t conn_id, bool client); + static BLEClient *getClientByID(uint16_t conn_id); + static BLEClient *getClientByAddress(BLEAddress address); static BLEClient *getClientByGattIf(uint16_t conn_id); static void setCustomGapHandler(gap_event_handler handler); + static void deinit(bool release_memory = false); + + /*************************************************************************** + * Bluedroid public declarations * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) + static void setEncryptionLevel(esp_ble_sec_act_t level); static void setCustomGattcHandler(gattc_event_handler handler); static void setCustomGattsHandler(gatts_event_handler handler); - static void deinit(bool release_memory = false); - static uint16_t m_localMTU; - static esp_ble_sec_act_t m_securityLevel; +#endif + + /*************************************************************************** + * NimBLE public declarations * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) + static void onReset(int reason); + static void onSync(void); + static void host_task(void *param); + static bool setOwnAddrType(uint8_t type); + static bool setOwnAddr(BLEAddress &addr); + static bool setOwnAddr(uint8_t *addr); + static void setDeviceCallbacks(BLEDeviceCallbacks *cb); + static bool onWhiteList(BLEAddress &address); +#endif private: + friend class BLEClient; + friend class BLEScan; + friend class BLEServer; + friend class BLECharacteristic; + friend class BLEAdvertising; + + /*************************************************************************** + * Common private properties * + ***************************************************************************/ + static BLEServer *m_pServer; static BLEScan *m_pScan; static BLEClient *m_pClient; static BLESecurityCallbacks *m_securityCallbacks; static BLEAdvertising *m_bleAdvertising; - static esp_gatt_if_t getGattcIF(); static std::map m_connectedClientsMap; static portMUX_TYPE mux; - static void gattClientEventHandler(esp_gattc_cb_event_t event, esp_gatt_if_t gattc_if, esp_ble_gattc_cb_param_t *param); + /*************************************************************************** + * NimBLE private properties * + ***************************************************************************/ - static void gattServerEventHandler(esp_gatts_cb_event_t event, esp_gatt_if_t gatts_if, esp_ble_gatts_cb_param_t *param); +#if defined(CONFIG_NIMBLE_ENABLED) + static uint8_t m_ownAddrType; + static bool m_synced; + static std::vector m_whiteList; + static BLEDeviceCallbacks defaultDeviceCallbacks; + static BLEDeviceCallbacks *m_pDeviceCallbacks; + static ble_gap_event_listener m_listener; +#endif + /*************************************************************************** + * Bluedroid private declarations * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) + static esp_gatt_if_t getGattcIF(); + static void gattClientEventHandler(esp_gattc_cb_event_t event, esp_gatt_if_t gattc_if, esp_ble_gattc_cb_param_t *param); + static void gattServerEventHandler(esp_gatts_cb_event_t event, esp_gatt_if_t gatts_if, esp_ble_gatts_cb_param_t *param); static void gapEventHandler(esp_gap_ble_cb_event_t event, esp_ble_gap_cb_param_t *param); +#endif +}; // class BLE -public: - /* custom gap and gatt handlers for flexibility */ - static gap_event_handler m_customGapHandler; - static gattc_event_handler m_customGattcHandler; - static gatts_event_handler m_customGattsHandler; +/*************************************************************************** + * NimBLE specific classes * + ***************************************************************************/ -}; // class BLE +#if defined(CONFIG_NIMBLE_ENABLED) +class BLEDeviceCallbacks { +public: + virtual ~BLEDeviceCallbacks(){}; + virtual int onStoreStatus(struct ble_store_status_event *event, void *arg); +}; +#endif -#endif /* CONFIG_BLUEDROID_ENABLED */ +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ #endif /* MAIN_BLEDevice_H_ */ diff --git a/libraries/BLE/src/BLEEddystoneTLM.cpp b/libraries/BLE/src/BLEEddystoneTLM.cpp index 1a301f09011..ca0fb41b1a2 100644 --- a/libraries/BLE/src/BLEEddystoneTLM.cpp +++ b/libraries/BLE/src/BLEEddystoneTLM.cpp @@ -8,12 +8,16 @@ * Fix time stamp (0.1 second resolution) * Fixes based on EddystoneTLM frame specification https://github.com/google/eddystone/blob/master/eddystone-tlm/tlm-plain.md * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on pcbreflux's and h2zero's work) + * Description: Added support for NimBLE */ + #include "soc/soc_caps.h" #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) #include #include #include "esp32-hal-log.h" @@ -176,5 +180,5 @@ void BLEEddystoneTLM::setTime(uint32_t tmil) { m_eddystoneData.tmil = tmil; } // setTime -#endif /* CONFIG_BLUEDROID_ENABLED */ +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ diff --git a/libraries/BLE/src/BLEEddystoneTLM.cppwithheadder b/libraries/BLE/src/BLEEddystoneTLM.cppwithheadder deleted file mode 100644 index 07002dbab1f..00000000000 --- a/libraries/BLE/src/BLEEddystoneTLM.cppwithheadder +++ /dev/null @@ -1,202 +0,0 @@ -/* - * BLEEddystoneTLM.cpp - * - * Created on: Mar 12, 2018 - * Author: pcbreflux - * Edited on: Mar 20, 2020 by beegee-tokyo - * Fix temperature value (8.8 fixed format) - * Fix time stamp (0.1 second resolution) - * Fixes based on EddystoneTLM frame specification https://github.com/google/eddystone/blob/master/eddystone-tlm/tlm-plain.md - * - */ -#include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) -#include -#include -#include "esp32-hal-log.h" -#include "BLEEddystoneTLM.h" - -static const char LOG_TAG[] = "BLEEddystoneTLM"; - -BLEEddystoneTLM::BLEEddystoneTLM() { - m_eddystoneData.frameType = EDDYSTONE_TLM_FRAME_TYPE; - m_eddystoneData.version = 0; - m_eddystoneData.volt = 3300; // 3300mV = 3.3V - m_eddystoneData.temp = (uint16_t) ((float) 23.00)/256; - m_eddystoneData.advCount = 0; - m_eddystoneData.tmil = 0; - _initHeadder(); -} // BLEEddystoneTLM - -BLEEddystoneTLM::BLEEddystoneTLM(BLEAdvertisedDevice *advertisedDevice){ - char* payload = (char*)advertisedDevice->getPayload(); - for(int i = 0; i < advertisedDevice->getPayloadLength(); ++i){ - if(payload[i] == 0x16 && advertisedDevice->getPayloadLength() >= i+2+sizeof(m_eddystoneData) && payload[i+1] == 0xAA && payload[i+2] == 0xFE && payload[i+3] == 0x20){ - log_d("Eddystone TLM data frame starting at byte [%d]", i+3); - setData(std::string(payload+i+3, sizeof(m_eddystoneData))); - break; - } - } - _initHeadder(); -} - -String BLEEddystoneTLM::getData() { - return String((char*) &m_eddystoneData, sizeof(m_eddystoneData)); -} // getData - -BLEUUID BLEEddystoneTLM::getUUID() { - return beaconUUID; -} // getUUID - -uint8_t BLEEddystoneTLM::getVersion() { - return m_eddystoneData.version; -} // getVersion - -uint16_t BLEEddystoneTLM::getVolt() { - return ENDIAN_CHANGE_U16(m_eddystoneData.volt); -} // getVolt - -float BLEEddystoneTLM::getTemp() { - return EDDYSTONE_TEMP_U16_TO_FLOAT(m_eddystoneData.temp); -} // getTemp - -uint16_t BLEEddystoneTLM::getRawTemp() { - return ENDIAN_CHANGE_U16(m_eddystoneData.temp); -} // getRawTemp - -uint32_t BLEEddystoneTLM::getCount() { - return ENDIAN_CHANGE_U32(m_eddystoneData.advCount); -} // getCount - -uint32_t BLEEddystoneTLM::getTime() { - return (ENDIAN_CHANGE_U32(m_eddystoneData.tmil)) / 10; -} // getTime - -String BLEEddystoneTLM::getFrame(){ - String frame(BLEHeadder); - frame += String((char*) &m_eddystoneData, sizeof(m_eddystoneData)); - log_d("Compiled frame of length %d Bytes", frame.length()); - for(int i = 0; i < frame.length(); ++i){ - log_d("[%d]=0x%02X",i, frame[i]); - } - return frame; -} // getServiceData - -String BLEEddystoneTLM::toString() { - String out = ""; - uint32_t rawsec = ENDIAN_CHANGE_U32(m_eddystoneData.tmil); - char val[12]; - - out += "Version "; // + std::string(m_eddystoneData.version); - snprintf(val, sizeof(val), "%d", m_eddystoneData.version); - out += val; - out += "\n"; - out += "Battery Voltage "; // + ENDIAN_CHANGE_U16(m_eddystoneData.volt); - snprintf(val, sizeof(val), "%d", ENDIAN_CHANGE_U16(m_eddystoneData.volt)); - out += val; - out += " mV\n"; - - out += "Temperature "; - snprintf(val, sizeof(val), "%.2f", ((int16_t)ENDIAN_CHANGE_U16(m_eddystoneData.temp)) / 256.0f); - out += val; - out += " C\n"; - - out += "Adv. Count "; - snprintf(val, sizeof(val), "%d", ENDIAN_CHANGE_U32(m_eddystoneData.advCount)); - out += val; - out += "\n"; - - out += "Time in seconds "; - snprintf(val, sizeof(val), "%d", rawsec/10); - out += val; - out += "\n"; - - out += "Time "; - - snprintf(val, sizeof(val), "%04d", rawsec / 864000); - out += val; - out += "."; - - snprintf(val, sizeof(val), "%02d", (rawsec / 36000) % 24); - out += val; - out += ":"; - - snprintf(val, sizeof(val), "%02d", (rawsec / 600) % 60); - out += val; - out += ":"; - - snprintf(val, sizeof(val), "%02d", (rawsec / 10) % 60); - out += val; - out += "\n"; - - return out; -} // toString - -/** - * Set the raw data for the beacon record. - * Example: - * uint8_t *payload = advertisedDevice.getPayload(); - * eddystoneTLM.setData(std::string((char*)payload+22, advertisedDevice.getPayloadLength() - 22)); - * Note: the offset 22 works for current implementation of example BLE_EddystoneTLM Beacon.ino, however - * the position is not static and it is programmers responsibility to align the data. - * Data frame: - * | Field || Len | Type | UUID | EddyStone TLM | - * | Offset || 0 | 1 | 2 | 4 | - * | Len || 1 B | 1 B | 2 B | 14 B | - * | Data || ?? | ?? | 0xAA | 0xFE | ??? | - * - * EddyStone TLM frame: - * | Field || Type | Version | Batt mV | Beacon temp | Cnt since boot | Time since boot | - * | Offset || 0 | 1 | 2 | 4 | 6 | 10 | - * | Len || 1 B | 1 B | 2 B | 2 B | 4 B | 4 B | - * | Data || 0x20 | ?? | ?? | ?? | ?? | ?? | | | | | | | | | - */ -void BLEEddystoneTLM::setData(std::string data) { - if (data.length() != sizeof(m_eddystoneData)) { - log_e("Unable to set the data ... length passed in was %d and expected %d", data.length(), sizeof(m_eddystoneData)); - return; - } - memcpy(&m_eddystoneData, data.data(), data.length()); -} // setData - -void BLEEddystoneTLM::setUUID(BLEUUID l_uuid) { - beaconUUID = l_uuid; -} // setUUID - -void BLEEddystoneTLM::setVersion(uint8_t version) { - m_eddystoneData.version = version; -} // setVersion - -// Set voltage in ESP32 native Big endian and convert it to little endian used for BLE Frame -void BLEEddystoneTLM::setVolt(uint16_t volt) { - m_eddystoneData.volt = ENDIAN_CHANGE_U16(volt); -} // setVolt - -void BLEEddystoneTLM::setTemp(float temp) { - m_eddystoneData.temp = EDDYSTONE_TEMP_FLOAT_TO_U16(temp); -} // setTemp - -void BLEEddystoneTLM::setCount(uint32_t advCount) { - m_eddystoneData.advCount = advCount; -} // setCount - -void BLEEddystoneTLM::setTime(uint32_t tmil) { - m_eddystoneData.tmil = tmil; -} // setTime - -void BLEEddystoneTLM::_initHeadder(){ - BLEHeadder[0] = 0x02; // Len - BLEHeadder[1] = 0x01; // Type Flags - BLEHeadder[2] = 0x06; // GENERAL_DISC_MODE 0x02 | BR_EDR_NOT_SUPPORTED 0x04 - BLEHeadder[3] = 0x03; // Len - BLEHeadder[4] = 0x03; // Type 16-Bit UUID - BLEHeadder[5] = 0xAA; // Eddystone UUID 2 -> 0xFEAA LSB - BLEHeadder[6] = 0xFE; // Eddystone UUID 1 MSB - BLEHeadder[7] = 0x11; // Length of TLM Beacon Data is constant 17 B (not counting the length field itself) - BLEHeadder[8] = 0x16; // Type Service Data - BLEHeadder[9] = 0xAA; // Eddystone UUID 2 -> 0xFEAA LSB - BLEHeadder[10] = 0xFE; // Eddystone UUID 1 MSB - BLEHeadder[11] = 0x20; // Eddystone Frame Type - TLM -} - -#endif diff --git a/libraries/BLE/src/BLEEddystoneTLM.h b/libraries/BLE/src/BLEEddystoneTLM.h index 3981af4a4a9..17915a670da 100644 --- a/libraries/BLE/src/BLEEddystoneTLM.h +++ b/libraries/BLE/src/BLEEddystoneTLM.h @@ -1,8 +1,12 @@ /* - * BLEEddystoneTLM.cpp + * BLEEddystoneTLM.h * * Created on: Mar 12, 2018 * Author: pcbreflux + * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on pcbreflux's and h2zero's work) + * Description: Added support for NimBLE */ #ifndef _BLEEddystoneTLM_H_ @@ -10,6 +14,9 @@ #include "soc/soc_caps.h" #if SOC_BLE_SUPPORTED +#include "sdkconfig.h" +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) + #include "BLEUUID.h" #include @@ -57,5 +64,6 @@ class BLEEddystoneTLM { } __attribute__((packed)) m_eddystoneData; }; // BLEEddystoneTLM +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ #endif /* _BLEEddystoneTLM_H_ */ diff --git a/libraries/BLE/src/BLEEddystoneURL.cpp b/libraries/BLE/src/BLEEddystoneURL.cpp index ddee8af0b30..495671ca49b 100644 --- a/libraries/BLE/src/BLEEddystoneURL.cpp +++ b/libraries/BLE/src/BLEEddystoneURL.cpp @@ -3,14 +3,20 @@ * * Created on: Mar 12, 2018 * Author: pcbreflux + * * Upgraded on: Feb 20, 2023 * By: Tomas Pilny + * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on pcbreflux's and h2zero's work) + * Description: Added support for NimBLE */ + #include "soc/soc_caps.h" #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) #include #include "esp32-hal-log.h" #include "BLEEddystoneURL.h" @@ -158,7 +164,11 @@ void BLEEddystoneURL::setData(String data) { } // setData void BLEEddystoneURL::setUUID(BLEUUID l_uuid) { +#if defined(CONFIG_BLUEDROID_ENABLED) uint16_t beaconUUID = l_uuid.getNative()->uuid.uuid16; +#elif defined(CONFIG_NIMBLE_ENABLED) + uint16_t beaconUUID = l_uuid.getNative()->u16.value; +#endif BLEHeadder[10] = beaconUUID >> 8; BLEHeadder[9] = beaconUUID & 0x00FF; } // setUUID diff --git a/libraries/BLE/src/BLEEddystoneURL.h b/libraries/BLE/src/BLEEddystoneURL.h index 92668eb6855..9ed89a23694 100644 --- a/libraries/BLE/src/BLEEddystoneURL.h +++ b/libraries/BLE/src/BLEEddystoneURL.h @@ -3,9 +3,13 @@ * * Created on: Mar 12, 2018 * Author: pcbreflux + * * Upgraded on: Feb 20, 2023 * By: Tomas Pilny * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on pcbreflux's and h2zero's work) + * Description: Added support for NimBLE */ #ifndef _BLEEddystoneURL_H_ @@ -13,6 +17,9 @@ #include "soc/soc_caps.h" #if SOC_BLE_SUPPORTED +#include "sdkconfig.h" +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) + #include "BLEUUID.h" #include #include "esp_bt.h" @@ -57,5 +64,6 @@ class BLEEddystoneURL { char BLEHeadder[12]; }; // BLEEddystoneURL +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ #endif /* _BLEEddystoneURL_H_ */ diff --git a/libraries/BLE/src/BLEEddystoneURL.h.orig b/libraries/BLE/src/BLEEddystoneURL.h.orig deleted file mode 100644 index 57722d0b769..00000000000 --- a/libraries/BLE/src/BLEEddystoneURL.h.orig +++ /dev/null @@ -1,66 +0,0 @@ -/* - * BLEEddystoneURL.cpp - * - * Created on: Mar 12, 2018 - * Author: pcbreflux - * Upgraded on: Feb 17, 2023 - * By: Tomas Pilny - * - */ - -#ifndef _BLEEddystoneURL_H_ -#define _BLEEddystoneURL_H_ -#include "BLEUUID.h" -#include -#include - -#define EDDYSTONE_URL_FRAME_TYPE 0x10 - -extern String EDDYSTONE_URL_PREFIX[]; -extern String EDDYSTONE_URL_SUFFIX[]; - -/** - * @brief Representation of a beacon. - * See: - * * https://github.com/google/eddystone - */ -class BLEEddystoneURL { -public: - BLEEddystoneURL(); - BLEEddystoneURL(BLEAdvertisedDevice *advertisedDevice); - std::string getData(); - String getFrame(); - BLEUUID getUUID(); - int8_t getPower(); - std::string getURL(); - String getPrefix(); - String getSuffix(); - std::string getDecodedURL(); - void setData(std::string data); - void setUUID(BLEUUID l_uuid); - void setPower(int8_t advertisedTxPower); - void setURL(std::string url); - int setSmartURL(String url); - -private: -<<<<<<< Updated upstream - uint16_t beaconUUID; - uint8_t lengthURL; - struct { - uint8_t frameType; - int8_t advertisedTxPower; - uint8_t url[18]; // 18 bytes: 1 byte for URL scheme + up to 17 bytes of URL - } __attribute__((packed)) m_eddystoneData; - -======= - uint8_t lengthURL; // Describes length of URL part including prefix and suffix - max 18 B (excluding TX power, frame type and preceding header) - struct { - int8_t advertisedTxPower; - uint8_t url[18]; // Byte [0] is for prefix. Last byte **can** contain suffix - } __attribute__((packed)) m_eddystoneData; - void _initHeadder(); - char BLEHeadder[12]; ->>>>>>> Stashed changes -}; // BLEEddystoneURL - -#endif /* _BLEEddystoneURL_H_ */ diff --git a/libraries/BLE/src/BLEExceptions.cpp b/libraries/BLE/src/BLEExceptions.cpp index 4e6c31fca22..b88ea337493 100644 --- a/libraries/BLE/src/BLEExceptions.cpp +++ b/libraries/BLE/src/BLEExceptions.cpp @@ -5,4 +5,9 @@ * Author: kolban */ +#include "soc/soc_caps.h" +#if SOC_BLE_SUPPORTED + //#include "BLEExceptions.h" + +#endif /* SOC_BLE_SUPPORTED */ diff --git a/libraries/BLE/src/BLEHIDDevice.cpp b/libraries/BLE/src/BLEHIDDevice.cpp index 0873aa1049f..a255879e2d8 100644 --- a/libraries/BLE/src/BLEHIDDevice.cpp +++ b/libraries/BLE/src/BLEHIDDevice.cpp @@ -3,15 +3,37 @@ * * Created on: Jan 03, 2018 * Author: chegewara + * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on kolban's and h2zero's work) + * Description: Added support for NimBLE */ + #include "soc/soc_caps.h" #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) + +/*************************************************************************** + * Common includes * + ***************************************************************************/ #include "BLEHIDDevice.h" #include "BLE2904.h" +#include "BLEDescriptor.h" + +/*************************************************************************** + * NimBLE includes and definitions * + ***************************************************************************/ + +#ifdef CONFIG_NIMBLE_ENABLED +#include +#endif + +/*************************************************************************** + * Common functions * + ***************************************************************************/ BLEHIDDevice::BLEHIDDevice(BLEServer *server) { /* @@ -45,10 +67,12 @@ BLEHIDDevice::BLEHIDDevice(BLEServer *server) { m_batteryLevelCharacteristic = m_batteryService->createCharacteristic((uint16_t)0x2a19, BLECharacteristic::PROPERTY_READ | BLECharacteristic::PROPERTY_NOTIFY); m_batteryLevelCharacteristic->addDescriptor(batteryLevelDescriptor); +#if CONFIG_BLUEDROID_ENABLED BLE2902 *batLevelIndicator = new BLE2902(); // Battery Level Notification is ON by default, making it work always on BLE Pairing and Bonding batLevelIndicator->setNotifications(true); m_batteryLevelCharacteristic->addDescriptor(batLevelIndicator); +#endif /* * This value is setup here because its default value in most usage cases, its very rare to use boot mode @@ -117,16 +141,19 @@ BLECharacteristic *BLEHIDDevice::inputReport(uint8_t reportID) { BLECharacteristic *inputReportCharacteristic = m_hidService->createCharacteristic((uint16_t)0x2a4d, BLECharacteristic::PROPERTY_READ | BLECharacteristic::PROPERTY_NOTIFY); BLEDescriptor *inputReportDescriptor = new BLEDescriptor(BLEUUID((uint16_t)0x2908)); - BLE2902 *p2902 = new BLE2902(); inputReportCharacteristic->setAccessPermissions(ESP_GATT_PERM_READ_ENCRYPTED | ESP_GATT_PERM_WRITE_ENCRYPTED); inputReportDescriptor->setAccessPermissions(ESP_GATT_PERM_READ_ENCRYPTED | ESP_GATT_PERM_WRITE_ENCRYPTED); - p2902->setAccessPermissions(ESP_GATT_PERM_READ_ENCRYPTED | ESP_GATT_PERM_WRITE_ENCRYPTED); uint8_t desc1_val[] = {reportID, 0x01}; inputReportDescriptor->setValue((uint8_t *)desc1_val, 2); - inputReportCharacteristic->addDescriptor(p2902); inputReportCharacteristic->addDescriptor(inputReportDescriptor); +#if CONFIG_BLUEDROID_ENABLED + BLE2902 *p2902 = new BLE2902(); + p2902->setAccessPermissions(ESP_GATT_PERM_READ_ENCRYPTED | ESP_GATT_PERM_WRITE_ENCRYPTED); + inputReportCharacteristic->addDescriptor(p2902); +#endif + return inputReportCharacteristic; } @@ -175,7 +202,9 @@ BLECharacteristic *BLEHIDDevice::featureReport(uint8_t reportID) { */ BLECharacteristic *BLEHIDDevice::bootInput() { BLECharacteristic *bootInputCharacteristic = m_hidService->createCharacteristic((uint16_t)0x2a22, BLECharacteristic::PROPERTY_NOTIFY); +#if CONFIG_BLUEDROID_ENABLED bootInputCharacteristic->addDescriptor(new BLE2902()); +#endif return bootInputCharacteristic; } @@ -252,5 +281,5 @@ BLEService *BLEHIDDevice::batteryService() { return m_batteryService; } -#endif /* CONFIG_BLUEDROID_ENABLED */ +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ diff --git a/libraries/BLE/src/BLEHIDDevice.h b/libraries/BLE/src/BLEHIDDevice.h index a92a23c21d5..9dde9452c12 100644 --- a/libraries/BLE/src/BLEHIDDevice.h +++ b/libraries/BLE/src/BLEHIDDevice.h @@ -3,6 +3,10 @@ * * Created on: Jan 03, 2018 * Author: chegewara + * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on kolban's and h2zero's work) + * Description: Added support for NimBLE */ #ifndef _BLEHIDDEVICE_H_ @@ -12,7 +16,7 @@ #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) #include "BLECharacteristic.h" #include "BLEService.h" @@ -75,6 +79,6 @@ class BLEHIDDevice { BLECharacteristic *m_batteryLevelCharacteristic; //0x2a19 }; -#endif /* CONFIG_BLUEDROID_ENABLED */ +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ #endif /* _BLEHIDDEVICE_H_ */ diff --git a/libraries/BLE/src/BLERemoteCharacteristic.cpp b/libraries/BLE/src/BLERemoteCharacteristic.cpp index 60d5108c1fc..aec9500d6f3 100644 --- a/libraries/BLE/src/BLERemoteCharacteristic.cpp +++ b/libraries/BLE/src/BLERemoteCharacteristic.cpp @@ -3,46 +3,55 @@ * * Created on: Jul 8, 2017 * Author: kolban + * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on kolban's and h2zero's work) + * Description: Added support for NimBLE */ -#include "BLERemoteCharacteristic.h" - #include "soc/soc_caps.h" #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) + +/*************************************************************************** + * Common includes * + ***************************************************************************/ -#include #include #include +#include "WString.h" //#include "BLEExceptions.h" #include "BLEUtils.h" #include "GeneralUtils.h" +#include "BLERemoteCharacteristic.h" #include "BLERemoteDescriptor.h" #include "esp32-hal-log.h" -/** - * @brief Constructor. - * @param [in] handle The BLE server side handle of this characteristic. - * @param [in] uuid The UUID of this characteristic. - * @param [in] charProp The properties of this characteristic. - * @param [in] pRemoteService A reference to the remote service to which this remote characteristic pertains. - */ -BLERemoteCharacteristic::BLERemoteCharacteristic(uint16_t handle, BLEUUID uuid, esp_gatt_char_prop_t charProp, BLERemoteService *pRemoteService) { - log_v(">> BLERemoteCharacteristic: handle: %d 0x%d, uuid: %s", handle, handle, uuid.toString().c_str()); - m_handle = handle; - m_uuid = uuid; - m_charProp = charProp; - m_pRemoteService = pRemoteService; - m_notifyCallback = nullptr; - m_rawData = nullptr; - m_auth = ESP_GATT_AUTH_REQ_NONE; +/*************************************************************************** + * Bluedroid includes * + ***************************************************************************/ - retrieveDescriptors(); // Get the descriptors for this characteristic - log_v("<< BLERemoteCharacteristic"); -} // BLERemoteCharacteristic +#if defined(CONFIG_BLUEDROID_ENABLED) +#include +#endif + +/*************************************************************************** + * NimBLE includes * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) +#include +#include +#include +#include +#endif + +/*************************************************************************** + * Common functions * + ***************************************************************************/ /** *@brief Destructor. @@ -100,211 +109,6 @@ bool BLERemoteCharacteristic::canWriteNoResponse() { return (m_charProp & ESP_GATT_CHAR_PROP_BIT_WRITE_NR) != 0; } // canWriteNoResponse -/* -static bool compareSrvcId(esp_gatt_srvc_id_t id1, esp_gatt_srvc_id_t id2) { - if (id1.id.inst_id != id2.id.inst_id) { - return false; - } - if (!BLEUUID(id1.id.uuid).equals(BLEUUID(id2.id.uuid))) { - return false; - } - return true; -} // compareSrvcId -*/ - -/* -static bool compareGattId(esp_gatt_id_t id1, esp_gatt_id_t id2) { - if (id1.inst_id != id2.inst_id) { - return false; - } - if (!BLEUUID(id1.uuid).equals(BLEUUID(id2.uuid))) { - return false; - } - return true; -} // compareCharId -*/ - -/** - * @brief Handle GATT Client events. - * When an event arrives for a GATT client we give this characteristic the opportunity to - * take a look at it to see if there is interest in it. - * @param [in] event The type of event. - * @param [in] gattc_if The interface on which the event was received. - * @param [in] evtParam Payload data for the event. - * @returns N/A - */ -void BLERemoteCharacteristic::gattClientEventHandler(esp_gattc_cb_event_t event, esp_gatt_if_t gattc_if, esp_ble_gattc_cb_param_t *evtParam) { - switch (event) { - // ESP_GATTC_NOTIFY_EVT - // - // notify - // - uint16_t conn_id - The connection identifier of the server. - // - esp_bd_addr_t remote_bda - The device address of the BLE server. - // - uint16_t handle - The handle of the characteristic for which the event is being received. - // - uint16_t value_len - The length of the received data. - // - uint8_t* value - The received data. - // - bool is_notify - True if this is a notify, false if it is an indicate. - // - // We have received a notification event which means that the server wishes us to know about a notification - // piece of data. What we must now do is find the characteristic with the associated handle and then - // invoke its notification callback (if it has one). - case ESP_GATTC_NOTIFY_EVT: - { - if (evtParam->notify.handle != getHandle()) { - break; - } - if (m_notifyCallback != nullptr) { - log_d("Invoking callback for notification on characteristic %s", toString().c_str()); - m_notifyCallback(this, evtParam->notify.value, evtParam->notify.value_len, evtParam->notify.is_notify); - } // End we have a callback function ... - break; - } // ESP_GATTC_NOTIFY_EVT - - // ESP_GATTC_READ_CHAR_EVT - // This event indicates that the server has responded to the read request. - // - // read: - // - esp_gatt_status_t status - // - uint16_t conn_id - // - uint16_t handle - // - uint8_t* value - // - uint16_t value_len - case ESP_GATTC_READ_CHAR_EVT: - { - // If this event is not for us, then nothing further to do. - if (evtParam->read.handle != getHandle()) { - break; - } - - // At this point, we have determined that the event is for us, so now we save the value - // and unlock the semaphore to ensure that the requester of the data can continue. - if (evtParam->read.status == ESP_GATT_OK) { - m_value = String((char *)evtParam->read.value, evtParam->read.value_len); - if (m_rawData != nullptr) { - free(m_rawData); - } - m_rawData = (uint8_t *)calloc(evtParam->read.value_len, sizeof(uint8_t)); - memcpy(m_rawData, evtParam->read.value, evtParam->read.value_len); - } else { - m_value = ""; - } - - m_semaphoreReadCharEvt.give(); - break; - } // ESP_GATTC_READ_CHAR_EVT - - // ESP_GATTC_REG_FOR_NOTIFY_EVT - // - // reg_for_notify: - // - esp_gatt_status_t status - // - uint16_t handle - case ESP_GATTC_REG_FOR_NOTIFY_EVT: - { - // If the request is not for this BLERemoteCharacteristic then move on to the next. - if (evtParam->reg_for_notify.handle != getHandle()) { - break; - } - - // We have processed the notify registration and can unlock the semaphore. - m_semaphoreRegForNotifyEvt.give(); - break; - } // ESP_GATTC_REG_FOR_NOTIFY_EVT - - // ESP_GATTC_UNREG_FOR_NOTIFY_EVT - // - // unreg_for_notify: - // - esp_gatt_status_t status - // - uint16_t handle - case ESP_GATTC_UNREG_FOR_NOTIFY_EVT: - { - if (evtParam->unreg_for_notify.handle != getHandle()) { - break; - } - // We have processed the notify un-registration and can unlock the semaphore. - m_semaphoreRegForNotifyEvt.give(); - break; - } // ESP_GATTC_UNREG_FOR_NOTIFY_EVT: - - // ESP_GATTC_WRITE_CHAR_EVT - // - // write: - // - esp_gatt_status_t status - // - uint16_t conn_id - // - uint16_t handle - case ESP_GATTC_WRITE_CHAR_EVT: - { - // Determine if this event is for us and, if not, pass onwards. - if (evtParam->write.handle != getHandle()) { - break; - } - - // There is nothing further we need to do here. This is merely an indication - // that the write has completed and we can unlock the caller. - m_semaphoreWriteCharEvt.give(); - break; - } // ESP_GATTC_WRITE_CHAR_EVT - - case ESP_GATTC_READ_DESCR_EVT: - case ESP_GATTC_WRITE_DESCR_EVT: - for (auto &myPair : m_descriptorMap) { - myPair.second->gattClientEventHandler(event, gattc_if, evtParam); - } - break; - - case ESP_GATTC_DISCONNECT_EVT: - // Cleanup semaphores to avoid deadlocks. - m_semaphoreReadCharEvt.give(1); - m_semaphoreWriteCharEvt.give(1); - break; - - default: break; - } // End switch -}; // gattClientEventHandler - -/** - * @brief Populate the descriptors (if any) for this characteristic. - */ -void BLERemoteCharacteristic::retrieveDescriptors() { - log_v(">> retrieveDescriptors() for characteristic: %s", getUUID().toString().c_str()); - - removeDescriptors(); // Remove any existing descriptors. - - // Loop over each of the descriptors within the service associated with this characteristic. - // For each descriptor we find, create a BLERemoteDescriptor instance. - uint16_t offset = 0; - esp_gattc_descr_elem_t result; - while (true) { - uint16_t count = 10; - esp_gatt_status_t status = ::esp_ble_gattc_get_all_descr( - getRemoteService()->getClient()->getGattcIf(), getRemoteService()->getClient()->getConnId(), getHandle(), &result, &count, offset - ); - - if (status == ESP_GATT_INVALID_OFFSET) { // We have reached the end of the entries. - break; - } - - if (status != ESP_GATT_OK) { - log_e("esp_ble_gattc_get_all_descr: %s", BLEUtils::gattStatusToString(status).c_str()); - break; - } - - if (count == 0) { - break; - } - - log_d("Found a descriptor: Handle: %d, UUID: %s", result.handle, BLEUUID(result.uuid).toString().c_str()); - - // We now have a new characteristic ... let us add that to our set of known characteristics - BLERemoteDescriptor *pNewRemoteDescriptor = new BLERemoteDescriptor(result.handle, BLEUUID(result.uuid), this); - - m_descriptorMap.insert(std::pair(pNewRemoteDescriptor->getUUID().toString().c_str(), pNewRemoteDescriptor)); - - offset++; - } // while true - //m_haveCharacteristics = true; // Remember that we have received the characteristics. - log_v("<< retrieveDescriptors(): Found %d descriptors.", offset); -} // getDescriptors - /** * @brief Retrieve the map of descriptors keyed by UUID. */ @@ -404,44 +208,6 @@ float BLERemoteCharacteristic::readFloat() { return 0.0; } // readFloat -/** - * @brief Read the value of the remote characteristic. - * @return The value of the remote characteristic. - */ -String BLERemoteCharacteristic::readValue() { - log_v(">> readValue(): uuid: %s, handle: %d 0x%.2x", getUUID().toString().c_str(), getHandle(), getHandle()); - - // Check to see that we are connected. - if (!getRemoteService()->getClient()->isConnected()) { - log_e("Disconnected"); - return String(); - } - - m_semaphoreReadCharEvt.take("readValue"); - - // Ask the BLE subsystem to retrieve the value for the remote hosted characteristic. - // This is an asynchronous request which means that we must block waiting for the response - // to become available. - esp_err_t errRc = ::esp_ble_gattc_read_char( - m_pRemoteService->getClient()->getGattcIf(), - m_pRemoteService->getClient()->getConnId(), // The connection ID to the BLE server - getHandle(), // The handle of this characteristic - m_auth - ); // Security - - if (errRc != ESP_OK) { - log_e("esp_ble_gattc_read_char: rc=%d %s", errRc, GeneralUtils::errorToString(errRc)); - return ""; - } - - // Block waiting for the event that indicates that the read has completed. When it has, the String found - // in m_value will contain our data. - m_semaphoreReadCharEvt.wait("readValue"); - - log_v("<< readValue(): length: %d", m_value.length()); - return m_value; -} // readValue - /** * @brief Register for notifications. * @param [in] notifyCallback A callback to be invoked for a notification. If NULL is provided then we are @@ -451,13 +217,14 @@ String BLERemoteCharacteristic::readValue() { void BLERemoteCharacteristic::registerForNotify(notify_callback notifyCallback, bool notifications, bool descriptorRequiresRegistration) { log_v(">> registerForNotify(): %s", toString().c_str()); +#if defined(CONFIG_BLUEDROID_ENABLED) m_notifyCallback = notifyCallback; // Save the notification callback. m_semaphoreRegForNotifyEvt.take("registerForNotify"); if (notifyCallback != nullptr) { // If we have a callback function, then this is a registration. esp_err_t errRc = ::esp_ble_gattc_register_for_notify( - m_pRemoteService->getClient()->getGattcIf(), *m_pRemoteService->getClient()->getPeerAddress().getNative(), getHandle() + m_pRemoteService->getClient()->getGattcIf(), m_pRemoteService->getClient()->getPeerAddress().getNative(), getHandle() ); if (errRc != ESP_OK) { @@ -475,7 +242,7 @@ void BLERemoteCharacteristic::registerForNotify(notify_callback notifyCallback, } // End Register else { // If we weren't passed a callback function, then this is an unregistration. esp_err_t errRc = ::esp_ble_gattc_unregister_for_notify( - m_pRemoteService->getClient()->getGattcIf(), *m_pRemoteService->getClient()->getPeerAddress().getNative(), getHandle() + m_pRemoteService->getClient()->getGattcIf(), m_pRemoteService->getClient()->getPeerAddress().getNative(), getHandle() ); if (errRc != ESP_OK) { @@ -491,8 +258,22 @@ void BLERemoteCharacteristic::registerForNotify(notify_callback notifyCallback, m_semaphoreRegForNotifyEvt.wait("registerForNotify"); - log_v("<< registerForNotify()"); -} // registerForNotify +#endif + +#if defined(CONFIG_NIMBLE_ENABLED) + bool success; + if (notifyCallback != nullptr) { + success = subscribe(notifications, notifyCallback, descriptorRequiresRegistration); + } else { + success = unsubscribe(descriptorRequiresRegistration); + } + + if (!success) { + log_e("Failed to subscribe/unsubscribe for notify"); + } +#endif + log_v("<< registerForNotify()"); +} // registerForNotify /** * @brief Delete the descriptors in the descriptor map. @@ -532,8 +313,8 @@ String BLERemoteCharacteristic::toString() { * @param [in] response Do we expect a response? * @return N/A. */ -void BLERemoteCharacteristic::writeValue(String newValue, bool response) { - writeValue((uint8_t *)newValue.c_str(), newValue.length(), response); +bool BLERemoteCharacteristic::writeValue(String newValue, bool response) { + return writeValue((uint8_t *)newValue.c_str(), newValue.length(), response); } // writeValue /** @@ -544,58 +325,678 @@ void BLERemoteCharacteristic::writeValue(String newValue, bool response) { * @param [in] response Whether we require a response from the write. * @return N/A. */ -void BLERemoteCharacteristic::writeValue(uint8_t newValue, bool response) { - writeValue(&newValue, 1, response); +bool BLERemoteCharacteristic::writeValue(uint8_t newValue, bool response) { + return writeValue(&newValue, 1, response); } // writeValue +/** + * @brief Read raw data from remote characteristic as hex bytes + * @return return pointer data read + */ +uint8_t *BLERemoteCharacteristic::readRawData() { + return m_rawData; +} + +/** + * @brief Set authentication request type for characteristic + * @param [in] auth Authentication request type. + */ +void BLERemoteCharacteristic::setAuth(uint8_t auth) { + m_auth = auth; +} + +/*************************************************************************** + * Bluedroid functions * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) + +/** + * @brief Constructor. + * @param [in] handle The BLE server side handle of this characteristic. + * @param [in] uuid The UUID of this characteristic. + * @param [in] charProp The properties of this characteristic. + * @param [in] pRemoteService A reference to the remote service to which this remote characteristic pertains. + */ +BLERemoteCharacteristic::BLERemoteCharacteristic(uint16_t handle, BLEUUID uuid, esp_gatt_char_prop_t charProp, BLERemoteService *pRemoteService) { + log_v(">> BLERemoteCharacteristic: handle: %d 0x%d, uuid: %s", handle, handle, uuid.toString().c_str()); + m_handle = handle; + m_uuid = uuid; + m_charProp = charProp; + m_pRemoteService = pRemoteService; + m_notifyCallback = nullptr; + m_rawData = nullptr; + m_auth = ESP_GATT_AUTH_REQ_NONE; + + retrieveDescriptors(); // Get the descriptors for this characteristic + log_v("<< BLERemoteCharacteristic"); +} // BLERemoteCharacteristic + +/** + * @brief Handle GATT Client events. + * When an event arrives for a GATT client we give this characteristic the opportunity to + * take a look at it to see if there is interest in it. + * @param [in] event The type of event. + * @param [in] gattc_if The interface on which the event was received. + * @param [in] evtParam Payload data for the event. + * @returns N/A + */ +void BLERemoteCharacteristic::gattClientEventHandler(esp_gattc_cb_event_t event, esp_gatt_if_t gattc_if, esp_ble_gattc_cb_param_t *evtParam) { + switch (event) { + // ESP_GATTC_NOTIFY_EVT + // + // notify + // - uint16_t conn_id - The connection identifier of the server. + // - esp_bd_addr_t remote_bda - The device address of the BLE server. + // - uint16_t handle - The handle of the characteristic for which the event is being received. + // - uint16_t value_len - The length of the received data. + // - uint8_t* value - The received data. + // - bool is_notify - True if this is a notify, false if it is an indicate. + // + // We have received a notification event which means that the server wishes us to know about a notification + // piece of data. What we must now do is find the characteristic with the associated handle and then + // invoke its notification callback (if it has one). + case ESP_GATTC_NOTIFY_EVT: + { + if (evtParam->notify.handle != getHandle()) { + break; + } + if (m_notifyCallback != nullptr) { + log_d("Invoking callback for notification on characteristic %s", toString().c_str()); + m_notifyCallback(this, evtParam->notify.value, evtParam->notify.value_len, evtParam->notify.is_notify); + } // End we have a callback function ... + break; + } // ESP_GATTC_NOTIFY_EVT + + // ESP_GATTC_READ_CHAR_EVT + // This event indicates that the server has responded to the read request. + // + // read: + // - esp_gatt_status_t status + // - uint16_t conn_id + // - uint16_t handle + // - uint8_t* value + // - uint16_t value_len + case ESP_GATTC_READ_CHAR_EVT: + { + // If this event is not for us, then nothing further to do. + if (evtParam->read.handle != getHandle()) { + break; + } + + // At this point, we have determined that the event is for us, so now we save the value + // and unlock the semaphore to ensure that the requester of the data can continue. + if (evtParam->read.status == ESP_GATT_OK) { + m_value = String((char *)evtParam->read.value, evtParam->read.value_len); + if (m_rawData != nullptr) { + free(m_rawData); + } + m_rawData = (uint8_t *)calloc(evtParam->read.value_len, sizeof(uint8_t)); + memcpy(m_rawData, evtParam->read.value, evtParam->read.value_len); + } else { + m_value = ""; + } + + m_semaphoreReadCharEvt.give(); + break; + } // ESP_GATTC_READ_CHAR_EVT + + // ESP_GATTC_REG_FOR_NOTIFY_EVT + // + // reg_for_notify: + // - esp_gatt_status_t status + // - uint16_t handle + case ESP_GATTC_REG_FOR_NOTIFY_EVT: + { + // If the request is not for this BLERemoteCharacteristic then move on to the next. + if (evtParam->reg_for_notify.handle != getHandle()) { + break; + } + + // We have processed the notify registration and can unlock the semaphore. + m_semaphoreRegForNotifyEvt.give(); + break; + } // ESP_GATTC_REG_FOR_NOTIFY_EVT + + // ESP_GATTC_UNREG_FOR_NOTIFY_EVT + // + // unreg_for_notify: + // - esp_gatt_status_t status + // - uint16_t handle + case ESP_GATTC_UNREG_FOR_NOTIFY_EVT: + { + if (evtParam->unreg_for_notify.handle != getHandle()) { + break; + } + // We have processed the notify un-registration and can unlock the semaphore. + m_semaphoreRegForNotifyEvt.give(); + break; + } // ESP_GATTC_UNREG_FOR_NOTIFY_EVT: + + // ESP_GATTC_WRITE_CHAR_EVT + // + // write: + // - esp_gatt_status_t status + // - uint16_t conn_id + // - uint16_t handle + case ESP_GATTC_WRITE_CHAR_EVT: + { + // Determine if this event is for us and, if not, pass onwards. + if (evtParam->write.handle != getHandle()) { + break; + } + + // There is nothing further we need to do here. This is merely an indication + // that the write has completed and we can unlock the caller. + m_semaphoreWriteCharEvt.give(); + break; + } // ESP_GATTC_WRITE_CHAR_EVT + + case ESP_GATTC_READ_DESCR_EVT: + case ESP_GATTC_WRITE_DESCR_EVT: + for (auto &myPair : m_descriptorMap) { + myPair.second->gattClientEventHandler(event, gattc_if, evtParam); + } + break; + + case ESP_GATTC_DISCONNECT_EVT: + // Cleanup semaphores to avoid deadlocks. + m_semaphoreReadCharEvt.give(1); + m_semaphoreWriteCharEvt.give(1); + break; + + default: break; + } // End switch +}; // gattClientEventHandler + +/** + * @brief Populate the descriptors (if any) for this characteristic. + */ +void BLERemoteCharacteristic::retrieveDescriptors() { + log_v(">> retrieveDescriptors() for characteristic: %s", getUUID().toString().c_str()); + + removeDescriptors(); // Remove any existing descriptors. + + // Loop over each of the descriptors within the service associated with this characteristic. + // For each descriptor we find, create a BLERemoteDescriptor instance. + uint16_t offset = 0; + esp_gattc_descr_elem_t result; + while (true) { + uint16_t count = 10; + esp_gatt_status_t status = ::esp_ble_gattc_get_all_descr( + getRemoteService()->getClient()->getGattcIf(), getRemoteService()->getClient()->getConnId(), getHandle(), &result, &count, offset + ); + + if (status == ESP_GATT_INVALID_OFFSET) { // We have reached the end of the entries. + break; + } + + if (status != ESP_GATT_OK) { + log_e("esp_ble_gattc_get_all_descr: %s", BLEUtils::gattStatusToString(status).c_str()); + break; + } + + if (count == 0) { + break; + } + + log_d("Found a descriptor: Handle: %d, UUID: %s", result.handle, BLEUUID(result.uuid).toString().c_str()); + + // We now have a new characteristic ... let us add that to our set of known characteristics + BLERemoteDescriptor *pNewRemoteDescriptor = new BLERemoteDescriptor(result.handle, BLEUUID(result.uuid), this); + + m_descriptorMap.insert(std::pair(pNewRemoteDescriptor->getUUID().toString().c_str(), pNewRemoteDescriptor)); + + offset++; + } // while true + //m_haveCharacteristics = true; // Remember that we have received the characteristics. + log_v("<< retrieveDescriptors(): Found %d descriptors.", offset); +} // getDescriptors + +/** + * @brief Read the value of the remote characteristic. + * @return The value of the remote characteristic. + */ +String BLERemoteCharacteristic::readValue() { + log_v(">> readValue(): uuid: %s, handle: %d 0x%.2x", getUUID().toString().c_str(), getHandle(), getHandle()); + + // Check to see that we are connected. + if (!getRemoteService()->getClient()->isConnected()) { + log_e("Disconnected"); + return String(); + } + + m_semaphoreReadCharEvt.take("readValue"); + + // Ask the BLE subsystem to retrieve the value for the remote hosted characteristic. + // This is an asynchronous request which means that we must block waiting for the response + // to become available. + esp_err_t errRc = ::esp_ble_gattc_read_char( + m_pRemoteService->getClient()->getGattcIf(), + m_pRemoteService->getClient()->getConnId(), // The connection ID to the BLE server + getHandle(), // The handle of this characteristic + (esp_gatt_auth_req_t)m_auth + ); // Security + + if (errRc != ESP_OK) { + log_e("esp_ble_gattc_read_char: rc=%d %s", errRc, GeneralUtils::errorToString(errRc)); + return ""; + } + + // Block waiting for the event that indicates that the read has completed. When it has, the String found + // in m_value will contain our data. + m_semaphoreReadCharEvt.wait("readValue"); + + log_v("<< readValue(): length: %d", m_value.length()); + return m_value; +} // readValue + /** * @brief Write the new value for the characteristic from a data buffer. * @param [in] data A pointer to a data buffer. * @param [in] length The length of the data in the data buffer. * @param [in] response Whether we require a response from the write. + * @return True if successful */ -void BLERemoteCharacteristic::writeValue(uint8_t *data, size_t length, bool response) { +bool BLERemoteCharacteristic::writeValue(uint8_t *data, size_t length, bool response) { // writeValue(String((char*)data, length), response); log_v(">> writeValue(), length: %d", length); // Check to see that we are connected. if (!getRemoteService()->getClient()->isConnected()) { log_e("Disconnected"); - return; + return false; } m_semaphoreWriteCharEvt.take("writeValue"); // Invoke the ESP-IDF API to perform the write. esp_err_t errRc = ::esp_ble_gattc_write_char( m_pRemoteService->getClient()->getGattcIf(), m_pRemoteService->getClient()->getConnId(), getHandle(), length, data, - response ? ESP_GATT_WRITE_TYPE_RSP : ESP_GATT_WRITE_TYPE_NO_RSP, m_auth + response ? ESP_GATT_WRITE_TYPE_RSP : ESP_GATT_WRITE_TYPE_NO_RSP, (esp_gatt_auth_req_t)m_auth ); if (errRc != ESP_OK) { log_e("esp_ble_gattc_write_char: rc=%d %s", errRc, GeneralUtils::errorToString(errRc)); - return; + return false; } m_semaphoreWriteCharEvt.wait("writeValue"); log_v("<< writeValue"); + return true; } // writeValue +#endif + +/*************************************************************************** + * NimBLE functions * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) + /** - * @brief Read raw data from remote characteristic as hex bytes - * @return return pointer data read + * @brief Constructor. + * @param [in] reference to the service this characteristic belongs to. + * @param [in] ble_gatt_chr struct defined as: + * struct ble_gatt_chr { + * uint16_t def_handle; + * uint16_t val_handle; + * uint8_t properties; + * ble_uuid_any_t uuid; + * }; */ -uint8_t *BLERemoteCharacteristic::readRawData() { - return m_rawData; +BLERemoteCharacteristic::BLERemoteCharacteristic(BLERemoteService *pRemoteService, const struct ble_gatt_chr *chr) { + log_v(">> BLERemoteCharacteristic()"); + switch (chr->uuid.u.type) { + case BLE_UUID_TYPE_16: m_uuid = BLEUUID(chr->uuid.u16.value); break; + case BLE_UUID_TYPE_32: m_uuid = BLEUUID(chr->uuid.u32.value); break; + case BLE_UUID_TYPE_128: m_uuid = BLEUUID(const_cast(&chr->uuid.u128)); break; + default: break; + } + + m_handle = chr->val_handle; + m_defHandle = chr->def_handle; + m_endHandle = 0; + m_charProp = chr->properties; + m_pRemoteService = pRemoteService; + m_notifyCallback = nullptr; + m_rawData = nullptr; + m_auth = 0; + + retrieveDescriptors(); // Get the descriptors for this characteristic + + log_v("<< BLERemoteCharacteristic(): %s", m_uuid.toString().c_str()); +} // BLERemoteCharacteristic + +/** + * @brief Callback used by the API when a descriptor is discovered or search complete. + */ +int BLERemoteCharacteristic::descriptorDiscCB( + uint16_t conn_handle, const struct ble_gatt_error *error, uint16_t chr_val_handle, const struct ble_gatt_dsc *dsc, void *arg +) { + int rc = error->status; + log_d("Descriptor Discovered >> status: %d handle: %d", rc, (rc == 0) ? dsc->handle : -1); + + desc_filter_t *filter = (desc_filter_t *)arg; + const BLEUUID *uuid_filter = filter->uuid; + BLETaskData *pTaskData = (BLETaskData *)filter->task_data; + BLERemoteCharacteristic *characteristic = (BLERemoteCharacteristic *)pTaskData->m_pInstance; + + if (characteristic->getRemoteService()->getClient()->getConnId() != conn_handle) { + return 0; + } + + if (rc == 0 && characteristic->getHandle() == chr_val_handle && (!uuid_filter || ble_uuid_cmp(&uuid_filter->getNative()->u, &dsc->uuid.u) == 0)) { + BLERemoteDescriptor *pNewRemoteDescriptor = new BLERemoteDescriptor(characteristic, dsc); + characteristic->m_descriptorMap.insert( + std::pair(pNewRemoteDescriptor->getUUID().toString().c_str(), pNewRemoteDescriptor) + ); + rc = !!uuid_filter * BLE_HS_EDONE; + } + + if (rc != 0) { + BLEUtils::taskRelease(*pTaskData, rc); + log_d("<< Descriptor Discovery"); + } + + return rc; } /** - * @brief Set authentication request type for characteristic - * @param [in] auth Authentication request type. + * @brief Callback for characteristic read operation. + * @return success == 0 or error code. */ -void BLERemoteCharacteristic::setAuth(esp_gatt_auth_req_t auth) { - m_auth = auth; +int BLERemoteCharacteristic::onReadCB(uint16_t conn_handle, const struct ble_gatt_error *error, struct ble_gatt_attr *attr, void *arg) { + BLETaskData *pTaskData = static_cast(arg); + BLERemoteCharacteristic *characteristic = static_cast(pTaskData->m_pInstance); + + if (error->status == BLE_HS_ENOTCONN) { + log_e("<< Characteristic Read; Not connected"); + BLEUtils::taskRelease(*pTaskData, error->status); + return error->status; + } + + if (characteristic->getRemoteService()->getClient()->getConnId() != conn_handle) { + return 0; + } + + int rc = error->status; + log_i("Read complete; status=%d conn_handle=%d", rc, conn_handle); + + String *strBuf = (String *)pTaskData->m_pBuf; + + if (rc == 0) { + if (attr) { + uint32_t data_len = OS_MBUF_PKTLEN(attr->om); + if (((*strBuf).length() + data_len) > BLE_ATT_ATTR_MAX_LEN) { + rc = BLE_ATT_ERR_INVALID_ATTR_VALUE_LEN; + } else { + log_i("Got %d bytes", data_len); + (*strBuf) += String((char *)attr->om->om_data, data_len); + return 0; + } + } + } + + BLEUtils::taskRelease(*pTaskData, rc); + return rc; } -#endif /* CONFIG_BLUEDROID_ENABLED */ +/** + * @brief Callback for characteristic write operation. + * @return success == 0 or error code. + */ +int BLERemoteCharacteristic::onWriteCB(uint16_t conn_handle, const struct ble_gatt_error *error, struct ble_gatt_attr *attr, void *arg) { + BLETaskData *pTaskData = static_cast(arg); + BLERemoteCharacteristic *characteristic = static_cast(pTaskData->m_pInstance); + + if (error->status == BLE_HS_ENOTCONN) { + log_e("<< Characteristic Write; Not connected"); + BLEUtils::taskRelease(*pTaskData, error->status); + return error->status; + } + + if (characteristic->getRemoteService()->getClient()->getConnId() != conn_handle) { + return 0; + } + + log_i("Write complete; status=%d conn_handle=%d", error->status, conn_handle); + BLEUtils::taskRelease(*pTaskData, error->status); + return 0; +} + +/** + * @brief Populate the descriptors (if any) for this characteristic. + * @param [in] the end handle of the characteristic, or the service, whichever comes first. + */ +bool BLERemoteCharacteristic::retrieveDescriptors(const BLEUUID *uuid_filter) { + log_d(">> retrieveDescriptors() for characteristic: %s", getUUID().toString().c_str()); + + // If this is the last handle then there are no descriptors + if (m_handle == getRemoteService()->getEndHandle()) { + log_d("<< retrieveDescriptors(): No descriptors found"); + return true; + } + + BLETaskData taskData(const_cast(this)); + desc_filter_t filter = {uuid_filter, &taskData}; + int rc = 0; + + rc = ble_gattc_disc_all_dscs(getRemoteService()->getClient()->getConnId(), m_handle, m_endHandle, BLERemoteCharacteristic::descriptorDiscCB, &filter); + + if (rc != 0) { + log_e("ble_gattc_disc_all_dscs: rc=%d %s", rc, BLEUtils::returnCodeToString(rc)); + return false; + } + + [[maybe_unused]] + size_t prevDscCount = m_descriptorMap.size(); + BLEUtils::taskWait(taskData, BLE_NPL_TIME_FOREVER); + rc = ((BLETaskData *)filter.task_data)->m_flags; + + if (rc != BLE_HS_EDONE) { + log_e("<< retrieveDescriptors(): failed: rc=%d %s", rc, BLEUtils::returnCodeToString(rc)); + return false; + } + + log_d("<< retrieveDescriptors(): Found %d descriptors.", m_descriptorMap.size() - prevDscCount); + return true; +} // retrieveDescriptors + +/** + * @brief Read the value of the remote characteristic. + * @return The value of the remote characteristic. + */ +String BLERemoteCharacteristic::readValue() { + log_d(">> readValue(): uuid: %s, handle: %d 0x%.2x", getUUID().toString().c_str(), getHandle(), getHandle()); + + BLEClient *pClient = getRemoteService()->getClient(); + String value{}; + + if (!pClient->isConnected()) { + log_e("Disconnected"); + return value; + } + + int rc = 0; + int retryCount = 1; + BLETaskData taskData(const_cast(this), 0, &value); + + do { + rc = ble_gattc_read_long(pClient->getConnId(), m_handle, 0, BLERemoteCharacteristic::onReadCB, &taskData); + if (rc != 0) { + goto exit; + } + + BLEUtils::taskWait(taskData, BLE_NPL_TIME_FOREVER); + rc = taskData.m_flags; + + switch (rc) { + case 0: + case BLE_HS_EDONE: rc = 0; break; + // Characteristic is not long-readable, return with what we have. + case BLE_HS_ATT_ERR(BLE_ATT_ERR_ATTR_NOT_LONG): + log_i("Attribute not long"); + rc = ble_gattc_read(pClient->getConnId(), m_handle, BLERemoteCharacteristic::onReadCB, &taskData); + if (rc != 0) { + goto exit; + } + retryCount++; + break; + case BLE_HS_ATT_ERR(BLE_ATT_ERR_INSUFFICIENT_AUTHEN): + case BLE_HS_ATT_ERR(BLE_ATT_ERR_INSUFFICIENT_AUTHOR): + case BLE_HS_ATT_ERR(BLE_ATT_ERR_INSUFFICIENT_ENC): + if (retryCount && pClient->secureConnection()) { + break; + } + /* Else falls through. */ + default: goto exit; + } + } while (rc != 0 && retryCount--); + + m_semaphoreReadCharEvt.take("readValue"); + m_value = value; + m_rawData = (uint8_t *)calloc(value.length(), sizeof(uint8_t)); + for (size_t i = 0; i < value.length(); i++) { + m_rawData[i] = value[i]; + } + m_semaphoreReadCharEvt.give(); + +exit: + if (rc != 0) { + log_e("<< readValue failed rc=%d, %s", rc, BLEUtils::returnCodeToString(rc)); + } else { + log_d("<< readValue length: %d rc=%d", value.length(), rc); + } + + return value; +} // readValue + +/** + * @brief Write the new value for the characteristic from a data buffer. + * @param [in] data A pointer to a data buffer. + * @param [in] length The length of the data in the data buffer. + * @param [in] response Whether we require a response from the write. + * @return false if not connected or cant perform write for some reason. + */ +bool BLERemoteCharacteristic::writeValue(uint8_t *data, size_t length, bool response) { + log_d(">> writeValue(), length: %d", length); + + BLEClient *pClient = getRemoteService()->getClient(); + + if (!pClient->isConnected()) { + log_e("Disconnected"); + return false; + } + + int rc = 0; + int retryCount = 1; + uint16_t mtu = ble_att_mtu(pClient->getConnId()) - 3; + BLETaskData taskData(const_cast(this)); + + // Check if the data length is longer than we can write in one connection event. + // If so we must do a long write which requires a response. + if (length <= mtu && !response) { + rc = ble_gattc_write_no_rsp_flat(pClient->getConnId(), m_handle, data, length); + goto exit; + } + + do { + if (length > mtu) { + log_i("long write %d bytes", length); + os_mbuf *om = ble_hs_mbuf_from_flat(data, length); + rc = ble_gattc_write_long(pClient->getConnId(), m_handle, 0, om, BLERemoteCharacteristic::onWriteCB, &taskData); + } else { + rc = ble_gattc_write_flat(pClient->getConnId(), m_handle, data, length, BLERemoteCharacteristic::onWriteCB, &taskData); + } + if (rc != 0) { + goto exit; + } + + BLEUtils::taskWait(taskData, BLE_NPL_TIME_FOREVER); + rc = taskData.m_flags; + + switch (rc) { + case 0: + case BLE_HS_EDONE: rc = 0; break; + case BLE_HS_ATT_ERR(BLE_ATT_ERR_ATTR_NOT_LONG): + log_e("Long write not supported by peer; Truncating length to %d", mtu); + retryCount++; + length = mtu; + break; + + case BLE_HS_ATT_ERR(BLE_ATT_ERR_INSUFFICIENT_AUTHEN): + case BLE_HS_ATT_ERR(BLE_ATT_ERR_INSUFFICIENT_AUTHOR): + case BLE_HS_ATT_ERR(BLE_ATT_ERR_INSUFFICIENT_ENC): + if (retryCount && pClient->secureConnection()) { + break; + } + /* Else falls through. */ + default: goto exit; + } + } while (rc != 0 && retryCount--); + +exit: + if (rc != 0) { + log_e("<< writeValue failed rc=%d, %s", rc, BLEUtils::returnCodeToString(rc)); + } else { + log_d("<< writeValue success. length: %d rc=%d", length, rc); + } + + return (rc == 0); +} // writeValue + +/** + * @brief Subscribe or unsubscribe for notifications or indications. + * @param [in] val 0x00 to unsubscribe, 0x01 for notifications, 0x02 for indications. + * @param [in] notifyCallback A callback to be invoked for a notification. + * @param [in] response If write response required set this to true. + * If NULL is provided then no callback is performed. + * @return false if writing to the descriptor failed. + */ +bool BLERemoteCharacteristic::setNotify(uint16_t val, notify_callback notifyCallback, bool response) { + log_v(">> setNotify(): %s, %02x", toString().c_str(), val); + + m_notifyCallback = notifyCallback; + + BLERemoteDescriptor *desc = getDescriptor(BLEUUID((uint16_t)0x2902)); + if (desc == nullptr) { + log_w("<< setNotify(): Callback set, CCCD not found"); + return true; + } + + log_d("<< setNotify()"); + + response = true; // Always write with response as per Bluetooth core specification. + return desc->writeValue((uint8_t *)&val, 2, response); +} // setNotify + +/** + * @brief Subscribe for notifications or indications. + * @param [in] notifications If true, subscribe for notifications, false subscribe for indications. + * @param [in] notifyCallback A callback to be invoked for a notification. + * @param [in] response If true, require a write response from the descriptor write operation. + * If NULL is provided then no callback is performed. + * @return false if writing to the descriptor failed. + */ +bool BLERemoteCharacteristic::subscribe(bool notifications, notify_callback notifyCallback, bool response) { + if (notifications) { + return setNotify(0x01, notifyCallback, response); + } else { + return setNotify(0x02, notifyCallback, response); + } +} // subscribe + +/** + * @brief Unsubscribe for notifications or indications. + * @param [in] response bool if true, require a write response from the descriptor write operation. + * @return false if writing to the descriptor failed. + */ +bool BLERemoteCharacteristic::unsubscribe(bool response) { + return setNotify(0x00, nullptr, response); +} // unsubscribe + +#endif + +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ diff --git a/libraries/BLE/src/BLERemoteCharacteristic.h b/libraries/BLE/src/BLERemoteCharacteristic.h index dc63a3bc1a6..81ad7b2f4f5 100644 --- a/libraries/BLE/src/BLERemoteCharacteristic.h +++ b/libraries/BLE/src/BLERemoteCharacteristic.h @@ -3,6 +3,10 @@ * * Created on: Jul 8, 2017 * Author: kolban + * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on kolban's and h2zero's work) + * Description: Added support for NimBLE */ #ifndef COMPONENTS_CPP_UTILS_BLEREMOTECHARACTERISTIC_H_ @@ -11,27 +15,80 @@ #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) -#include +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) -#include +/*************************************************************************** + * Common includes * + ***************************************************************************/ +#include #include "BLERemoteService.h" #include "BLERemoteDescriptor.h" #include "BLEUUID.h" #include "RTOS.h" +#include "BLEUtils.h" + +/*************************************************************************** + * Bluedroid includes * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) +#include +#endif + +/*************************************************************************** + * NimBLE includes and definitions * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) +#include +#include +#include + +#define ESP_GATT_MAX_ATTR_LEN BLE_ATT_ATTR_MAX_LEN +#define ESP_GATT_CHAR_PROP_BIT_READ BLE_GATT_CHR_PROP_READ +#define ESP_GATT_CHAR_PROP_BIT_WRITE BLE_GATT_CHR_PROP_WRITE +#define ESP_GATT_CHAR_PROP_BIT_WRITE_NR BLE_GATT_CHR_PROP_WRITE_NO_RSP +#define ESP_GATT_CHAR_PROP_BIT_BROADCAST BLE_GATT_CHR_PROP_BROADCAST +#define ESP_GATT_CHAR_PROP_BIT_NOTIFY BLE_GATT_CHR_PROP_NOTIFY +#define ESP_GATT_CHAR_PROP_BIT_INDICATE BLE_GATT_CHR_PROP_INDICATE + +#endif + +/*************************************************************************** + * Common types * + ***************************************************************************/ + +typedef std::function notify_callback; + +/*************************************************************************** + * NimBLE types * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) +typedef struct { + const BLEUUID *uuid; + void *task_data; +} desc_filter_t; +#endif + +/*************************************************************************** + * Forward declarations * + ***************************************************************************/ class BLERemoteService; class BLERemoteDescriptor; -typedef std::function notify_callback; + /** * @brief A model of a remote %BLE characteristic. */ class BLERemoteCharacteristic { public: - ~BLERemoteCharacteristic(); + /*************************************************************************** + * Common public declarations * + ***************************************************************************/ - // Public member functions + ~BLERemoteCharacteristic(); bool canBroadcast(); bool canIndicate(); bool canNotify(); @@ -49,29 +106,34 @@ class BLERemoteCharacteristic { uint32_t readUInt32(); float readFloat(); void registerForNotify(notify_callback _callback, bool notifications = true, bool descriptorRequiresRegistration = true); - void writeValue(uint8_t *data, size_t length, bool response = false); - void writeValue(String newValue, bool response = false); - void writeValue(uint8_t newValue, bool response = false); + bool writeValue(uint8_t *data, size_t length, bool response = false); + bool writeValue(String newValue, bool response = false); + bool writeValue(uint8_t newValue, bool response = false); String toString(); uint8_t *readRawData(); - void setAuth(esp_gatt_auth_req_t auth); + void setAuth(uint8_t auth); + + /*************************************************************************** + * NimBLE public declarations * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) + bool subscribe(bool notifications = true, notify_callback notifyCallback = nullptr, bool response = true); + bool unsubscribe(bool response = true); +#endif private: - BLERemoteCharacteristic(uint16_t handle, BLEUUID uuid, esp_gatt_char_prop_t charProp, BLERemoteService *pRemoteService); friend class BLEClient; friend class BLERemoteService; friend class BLERemoteDescriptor; - // Private member functions - void gattClientEventHandler(esp_gattc_cb_event_t event, esp_gatt_if_t gattc_if, esp_ble_gattc_cb_param_t *evtParam); - - void removeDescriptors(); - void retrieveDescriptors(); + /*************************************************************************** + * Common private properties * + ***************************************************************************/ - // Private properties BLEUUID m_uuid; - esp_gatt_char_prop_t m_charProp; - esp_gatt_auth_req_t m_auth; + uint8_t m_charProp; + uint8_t m_auth; uint16_t m_handle; BLERemoteService *m_pRemoteService; FreeRTOS::Semaphore m_semaphoreReadCharEvt = FreeRTOS::Semaphore("ReadCharEvt"); @@ -83,8 +145,46 @@ class BLERemoteCharacteristic { // We maintain a map of descriptors owned by this characteristic keyed by a string representation of the UUID. std::map m_descriptorMap; + + /*************************************************************************** + * NimBLE private properties * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) + uint16_t m_defHandle; + uint16_t m_endHandle; +#endif + + /*************************************************************************** + * Common private declarations * + ***************************************************************************/ + + void removeDescriptors(); + + /*************************************************************************** + * Bluedroid private declarations * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) + BLERemoteCharacteristic(uint16_t handle, BLEUUID uuid, uint8_t charProp, BLERemoteService *pRemoteService); + void retrieveDescriptors(); + void gattClientEventHandler(esp_gattc_cb_event_t event, esp_gatt_if_t gattc_if, esp_ble_gattc_cb_param_t *evtParam); +#endif + + /*************************************************************************** + * NimBLE private declarations * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) + BLERemoteCharacteristic(BLERemoteService *pRemoteservice, const struct ble_gatt_chr *chr); + bool setNotify(uint16_t val, notify_callback notifyCallback = nullptr, bool response = true); + bool retrieveDescriptors(const BLEUUID *uuid_filter = nullptr); + static int onReadCB(uint16_t conn_handle, const struct ble_gatt_error *error, struct ble_gatt_attr *attr, void *arg); + static int onWriteCB(uint16_t conn_handle, const struct ble_gatt_error *error, struct ble_gatt_attr *attr, void *arg); + static int descriptorDiscCB(uint16_t conn_handle, const struct ble_gatt_error *error, uint16_t chr_val_handle, const struct ble_gatt_dsc *dsc, void *arg); +#endif }; // BLERemoteCharacteristic -#endif /* CONFIG_BLUEDROID_ENABLED */ +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ #endif /* COMPONENTS_CPP_UTILS_BLEREMOTECHARACTERISTIC_H_ */ diff --git a/libraries/BLE/src/BLERemoteDescriptor.cpp b/libraries/BLE/src/BLERemoteDescriptor.cpp index b6d654cf9ec..a142fe11880 100644 --- a/libraries/BLE/src/BLERemoteDescriptor.cpp +++ b/libraries/BLE/src/BLERemoteDescriptor.cpp @@ -3,23 +3,48 @@ * * Created on: Jul 8, 2017 * Author: kolban + * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on kolban's and h2zero's work) + * Description: Added support for NimBLE */ + #include "soc/soc_caps.h" #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) + +/*************************************************************************** + * Common includes * + ***************************************************************************/ + +#include "WString.h" #include #include "BLERemoteDescriptor.h" #include "GeneralUtils.h" #include "esp32-hal-log.h" -BLERemoteDescriptor::BLERemoteDescriptor(uint16_t handle, BLEUUID uuid, BLERemoteCharacteristic *pRemoteCharacteristic) { +/*************************************************************************** + * NimBLE includes * + ***************************************************************************/ +#if defined(CONFIG_NIMBLE_ENABLED) +#include +#include +#include +#include +#endif + +/*************************************************************************** + * Common functions * + ***************************************************************************/ + +BLERemoteDescriptor::BLERemoteDescriptor(uint16_t handle, BLEUUID uuid, BLERemoteCharacteristic *pRemoteCharacteristic) { m_handle = handle; m_uuid = uuid; m_pRemoteCharacteristic = pRemoteCharacteristic; - m_auth = ESP_GATT_AUTH_REQ_NONE; + m_auth = 0; } /** @@ -46,6 +71,75 @@ BLEUUID BLERemoteDescriptor::getUUID() { return m_uuid; } // getUUID +uint8_t BLERemoteDescriptor::readUInt8() { + String value = readValue(); + if (value.length() >= 1) { + return (uint8_t)value[0]; + } + return 0; +} // readUInt8 + +uint16_t BLERemoteDescriptor::readUInt16() { + String value = readValue(); + if (value.length() >= 2) { + return *(uint16_t *)value.c_str(); + } + return 0; +} // readUInt16 + +uint32_t BLERemoteDescriptor::readUInt32() { + String value = readValue(); + if (value.length() >= 4) { + return *(uint32_t *)value.c_str(); + } + return 0; +} // readUInt32 + +/** + * @brief Return a string representation of this BLE Remote Descriptor. + * @return A string representation of this BLE Remote Descriptor. + */ +String BLERemoteDescriptor::toString() { + char val[6]; + snprintf(val, sizeof(val), "%d", getHandle()); + String res = "handle: "; + res += val; + res += ", uuid: " + getUUID().toString(); + return res; +} // toString + +/** + * @brief Write data represented as a string to the BLE Remote Descriptor. + * @param [in] newValue The data to send to the remote descriptor. + * @param [in] response True if we expect a response. + */ +bool BLERemoteDescriptor::writeValue(String newValue, bool response) { + return writeValue((uint8_t *)newValue.c_str(), newValue.length(), response); +} // writeValue + +/** + * @brief Write a byte value to the Descriptor. + * @param [in] The single byte to write. + * @param [in] True if we expect a response. + */ +bool BLERemoteDescriptor::writeValue(uint8_t newValue, bool response) { + return writeValue(&newValue, 1, response); +} // writeValue + +/** + * @brief Set authentication request type for characteristic + * @param [in] auth Authentication request type. + */ +void BLERemoteDescriptor::setAuth(uint8_t auth) { + m_auth = auth; +} + +/*************************************************************************** + * Bluedroid functions * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) + void BLERemoteDescriptor::gattClientEventHandler(esp_gattc_cb_event_t event, esp_gatt_if_t gattc_if, esp_ble_gattc_cb_param_t *evtParam) { switch (event) { // ESP_GATTC_READ_DESCR_EVT @@ -99,7 +193,7 @@ String BLERemoteDescriptor::readValue() { m_pRemoteCharacteristic->getRemoteService()->getClient()->getGattcIf(), m_pRemoteCharacteristic->getRemoteService()->getClient()->getConnId(), // The connection ID to the BLE server getHandle(), // The handle of this characteristic - m_auth + (esp_gatt_auth_req_t)m_auth ); // Security if (errRc != ESP_OK) { @@ -115,55 +209,19 @@ String BLERemoteDescriptor::readValue() { return m_value; } // readValue -uint8_t BLERemoteDescriptor::readUInt8() { - String value = readValue(); - if (value.length() >= 1) { - return (uint8_t)value[0]; - } - return 0; -} // readUInt8 - -uint16_t BLERemoteDescriptor::readUInt16() { - String value = readValue(); - if (value.length() >= 2) { - return *(uint16_t *)value.c_str(); - } - return 0; -} // readUInt16 - -uint32_t BLERemoteDescriptor::readUInt32() { - String value = readValue(); - if (value.length() >= 4) { - return *(uint32_t *)value.c_str(); - } - return 0; -} // readUInt32 - -/** - * @brief Return a string representation of this BLE Remote Descriptor. - * @return A string representation of this BLE Remote Descriptor. - */ -String BLERemoteDescriptor::toString() { - char val[6]; - snprintf(val, sizeof(val), "%d", getHandle()); - String res = "handle: "; - res += val; - res += ", uuid: " + getUUID().toString(); - return res; -} // toString - /** * @brief Write data to the BLE Remote Descriptor. * @param [in] data The data to send to the remote descriptor. * @param [in] length The length of the data to send. * @param [in] response True if we expect a response. + * @return True if successful */ -void BLERemoteDescriptor::writeValue(uint8_t *data, size_t length, bool response) { +bool BLERemoteDescriptor::writeValue(uint8_t *data, size_t length, bool response) { log_v(">> writeValue: %s", toString().c_str()); // Check to see that we are connected. if (!getRemoteCharacteristic()->getRemoteService()->getClient()->isConnected()) { log_e("Disconnected"); - return; + return false; } m_semaphoreWriteDescrEvt.take("writeValue"); @@ -172,7 +230,7 @@ void BLERemoteDescriptor::writeValue(uint8_t *data, size_t length, bool response m_pRemoteCharacteristic->getRemoteService()->getClient()->getGattcIf(), m_pRemoteCharacteristic->getRemoteService()->getClient()->getConnId(), getHandle(), length, // Data length data, // Data - response ? ESP_GATT_WRITE_TYPE_RSP : ESP_GATT_WRITE_TYPE_NO_RSP, m_auth + response ? ESP_GATT_WRITE_TYPE_RSP : ESP_GATT_WRITE_TYPE_NO_RSP, (esp_gatt_auth_req_t)m_auth ); if (errRc != ESP_OK) { log_e("esp_ble_gattc_write_char_descr: %d", errRc); @@ -180,33 +238,246 @@ void BLERemoteDescriptor::writeValue(uint8_t *data, size_t length, bool response m_semaphoreWriteDescrEvt.wait("writeValue"); log_v("<< writeValue"); + return (errRc == ESP_OK); } // writeValue +#endif + +/*************************************************************************** + * NimBLE functions * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) + /** - * @brief Write data represented as a string to the BLE Remote Descriptor. - * @param [in] newValue The data to send to the remote descriptor. - * @param [in] response True if we expect a response. + * @brief Remote descriptor constructor. + * @param [in] pRemoteCharacteristic A pointer to the Characteristic that this belongs to. + * @param [in] dsc A pointer to the struct that contains the descriptor information. */ -void BLERemoteDescriptor::writeValue(String newValue, bool response) { - writeValue((uint8_t *)newValue.c_str(), newValue.length(), response); -} // writeValue +BLERemoteDescriptor::BLERemoteDescriptor(BLERemoteCharacteristic *pRemoteCharacteristic, const struct ble_gatt_dsc *dsc) { + log_d(">> BLERemoteDescriptor()"); + switch (dsc->uuid.u.type) { + case BLE_UUID_TYPE_16: m_uuid = BLEUUID(dsc->uuid.u16.value); break; + case BLE_UUID_TYPE_32: m_uuid = BLEUUID(dsc->uuid.u32.value); break; + case BLE_UUID_TYPE_128: m_uuid = BLEUUID(const_cast(&dsc->uuid.u128)); break; + default: break; + } + + m_handle = dsc->handle; + m_pRemoteCharacteristic = pRemoteCharacteristic; + m_auth = 0; + + log_d("<< BLERemoteDescriptor(): %s", m_uuid.toString().c_str()); +} /** - * @brief Write a byte value to the Descriptor. - * @param [in] The single byte to write. - * @param [in] True if we expect a response. + * @brief Read the value of the remote descriptor. + * @return The value of the remote descriptor. */ -void BLERemoteDescriptor::writeValue(uint8_t newValue, bool response) { - writeValue(&newValue, 1, response); -} // writeValue +String BLERemoteDescriptor::readValue() { + log_d(">> Descriptor readValue: %s", toString().c_str()); + + BLEClient *pClient = getRemoteCharacteristic()->getRemoteService()->getClient(); + String value{}; + + if (!pClient->isConnected()) { + log_e("Disconnected"); + return value; + } + + int rc = 0; + int retryCount = 1; + BLETaskData taskData(const_cast(this), 0, &value); + + do { + rc = ble_gattc_read_long(pClient->getConnId(), m_handle, 0, BLERemoteDescriptor::onReadCB, &taskData); + if (rc != 0) { + goto exit; + } + + BLEUtils::taskWait(taskData, BLE_NPL_TIME_FOREVER); + rc = taskData.m_flags; + + switch (rc) { + case 0: + case BLE_HS_EDONE: rc = 0; break; + // Descriptor is not long-readable, return with what we have. + case BLE_HS_ATT_ERR(BLE_ATT_ERR_ATTR_NOT_LONG): + log_i("Attribute not long"); + rc = ble_gattc_read(pClient->getConnId(), m_handle, BLERemoteDescriptor::onReadCB, &taskData); + if (rc != 0) { + goto exit; + } + retryCount++; + break; + case BLE_HS_ATT_ERR(BLE_ATT_ERR_INSUFFICIENT_AUTHEN): + case BLE_HS_ATT_ERR(BLE_ATT_ERR_INSUFFICIENT_AUTHOR): + case BLE_HS_ATT_ERR(BLE_ATT_ERR_INSUFFICIENT_ENC): + if (retryCount && pClient->secureConnection()) { + break; + } + /* Else falls through. */ + default: goto exit; + } + } while (rc != 0 && retryCount--); + + m_semaphoreReadDescrEvt.take("readValue"); + m_value = value; + m_semaphoreReadDescrEvt.give(); + +exit: + if (rc != 0) { + log_e("<< readValue failed rc=%d, %s", rc, BLEUtils::returnCodeToString(rc)); + } else { + log_d("<< Descriptor readValue(): length: %d rc=%d", value.length(), rc); + } + + return value; +} // readValue /** - * @brief Set authentication request type for characteristic - * @param [in] auth Authentication request type. + * @brief Callback for Descriptor read operation. + * @return success == 0 or error code. */ -void BLERemoteDescriptor::setAuth(esp_gatt_auth_req_t auth) { - m_auth = auth; +int BLERemoteDescriptor::onReadCB(uint16_t conn_handle, const struct ble_gatt_error *error, struct ble_gatt_attr *attr, void *arg) { + BLETaskData *pTaskData = static_cast(arg); + BLERemoteDescriptor *desc = static_cast(pTaskData->m_pInstance); + uint16_t conn_id = desc->getRemoteCharacteristic()->getRemoteService()->getClient()->getConnId(); + + if (error->status == BLE_HS_ENOTCONN) { + log_e("<< Descriptor Read; Not connected"); + BLEUtils::taskRelease(*pTaskData, error->status); + return error->status; + } + + if (conn_id != conn_handle) { + return 0; + } + + log_d("Read complete; status=%d conn_handle=%d", error->status, conn_handle); + + String *strBuf = static_cast(pTaskData->m_pBuf); + int rc = error->status; + + if (rc == 0) { + if (attr) { + uint32_t data_len = OS_MBUF_PKTLEN(attr->om); + if (((*strBuf).length() + data_len) > BLE_ATT_ATTR_MAX_LEN) { + rc = BLE_ATT_ERR_INVALID_ATTR_VALUE_LEN; + } else { + log_d("Got %d bytes", data_len); + (*strBuf) += String((char *)attr->om->om_data, data_len); + return 0; + } + } + } + + BLEUtils::taskRelease(*pTaskData, rc); + return rc; +} + +/** + * @brief Callback for descriptor write operation. + * @return success == 0 or error code. + */ +int BLERemoteDescriptor::onWriteCB(uint16_t conn_handle, const struct ble_gatt_error *error, struct ble_gatt_attr *attr, void *arg) { + BLETaskData *pTaskData = static_cast(arg); + BLERemoteDescriptor *descriptor = static_cast(pTaskData->m_pInstance); + int rc = error->status; + + if (rc == BLE_HS_ENOTCONN) { + log_e("<< Descriptor Write; Not connected"); + BLEUtils::taskRelease(*pTaskData, rc); + return rc; + } + + if (descriptor->getRemoteCharacteristic()->getRemoteService()->getClient()->getConnId() != conn_handle) { + return 0; + } + + log_i("Write complete; status=%d conn_handle=%d", rc, conn_handle); + + BLEUtils::taskRelease(*pTaskData, rc); + return 0; } -#endif /* CONFIG_BLUEDROID_ENABLED */ +/** + * @brief Write data to the BLE Remote Descriptor. + * @param [in] data The data to send to the remote descriptor. + * @param [in] length The length of the data to send. + * @param [in] response True if we expect a write response. + * @return True if successful + */ +bool BLERemoteDescriptor::writeValue(uint8_t *data, size_t length, bool response) { + log_d(">> Descriptor writeValue: %s", toString().c_str()); + + BLEClient *pClient = getRemoteCharacteristic()->getRemoteService()->getClient(); + + // Check to see that we are connected. + if (!pClient->isConnected()) { + log_e("Disconnected"); + return false; + } + + int rc = 0; + int retryCount = 1; + uint16_t mtu = ble_att_mtu(pClient->getConnId()) - 3; + BLETaskData taskData(const_cast(this)); + + // Check if the data length is longer than we can write in 1 connection event. + // If so we must do a long write which requires a response. + if (length <= mtu && !response) { + rc = ble_gattc_write_no_rsp_flat(pClient->getConnId(), m_handle, data, length); + goto exit; + } + + do { + if (length > mtu) { + log_i("long write %d bytes", length); + os_mbuf *om = ble_hs_mbuf_from_flat(data, length); + rc = ble_gattc_write_long(pClient->getConnId(), m_handle, 0, om, BLERemoteDescriptor::onWriteCB, &taskData); + } else { + rc = ble_gattc_write_flat(pClient->getConnId(), m_handle, data, length, BLERemoteDescriptor::onWriteCB, &taskData); + } + + if (rc != 0) { + goto exit; + } + + BLEUtils::taskWait(taskData, BLE_NPL_TIME_FOREVER); + rc = taskData.m_flags; + + switch (rc) { + case 0: + case BLE_HS_EDONE: rc = 0; break; + case BLE_HS_ATT_ERR(BLE_ATT_ERR_ATTR_NOT_LONG): + log_e("Long write not supported by peer; Truncating length to %d", mtu); + retryCount++; + length = mtu; + break; + + case BLE_HS_ATT_ERR(BLE_ATT_ERR_INSUFFICIENT_AUTHEN): + case BLE_HS_ATT_ERR(BLE_ATT_ERR_INSUFFICIENT_AUTHOR): + case BLE_HS_ATT_ERR(BLE_ATT_ERR_INSUFFICIENT_ENC): + if (retryCount && pClient->secureConnection()) { + break; + } + /* Else falls through. */ + default: goto exit; + } + } while (rc != 0 && retryCount--); + +exit: + if (rc != 0) { + log_e("<< writeValue failed rc=%d, %s", rc, BLEUtils::returnCodeToString(rc)); + } else { + log_d("<< writeValue success. length: %d rc=%d", length, rc); + } + + return (rc == 0); +} // writeValue + +#endif + +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ diff --git a/libraries/BLE/src/BLERemoteDescriptor.h b/libraries/BLE/src/BLERemoteDescriptor.h index 94b11f1490a..afe113df551 100644 --- a/libraries/BLE/src/BLERemoteDescriptor.h +++ b/libraries/BLE/src/BLERemoteDescriptor.h @@ -3,6 +3,10 @@ * * Created on: Jul 8, 2017 * Author: kolban + * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on kolban's and h2zero's work) + * Description: Added support for NimBLE */ #ifndef COMPONENTS_CPP_UTILS_BLEREMOTEDESCRIPTOR_H_ @@ -11,21 +15,41 @@ #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) -#include +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) -#include +/*************************************************************************** + * Common includes * + ***************************************************************************/ +#include #include "BLERemoteCharacteristic.h" #include "BLEUUID.h" #include "RTOS.h" +#include "BLEUtils.h" + +/*************************************************************************** + * Bluedroid includes * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) +#include +#endif + +/*************************************************************************** + * Forward declarations * + ***************************************************************************/ class BLERemoteCharacteristic; + /** * @brief A model of remote %BLE descriptor. */ class BLERemoteDescriptor { public: + /*************************************************************************** + * Common public declarations * + ***************************************************************************/ + uint16_t getHandle(); BLERemoteCharacteristic *getRemoteCharacteristic(); BLEUUID getUUID(); @@ -34,24 +58,50 @@ class BLERemoteDescriptor { uint16_t readUInt16(void); uint32_t readUInt32(void); String toString(void); - void writeValue(uint8_t *data, size_t length, bool response = false); - void writeValue(String newValue, bool response = false); - void writeValue(uint8_t newValue, bool response = false); - void setAuth(esp_gatt_auth_req_t auth); + bool writeValue(uint8_t *data, size_t length, bool response = false); + bool writeValue(String newValue, bool response = false); + bool writeValue(uint8_t newValue, bool response = false); + void setAuth(uint8_t auth); + + /*************************************************************************** + * Bluedroid public declarations * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) void gattClientEventHandler(esp_gattc_cb_event_t event, esp_gatt_if_t gattc_if, esp_ble_gattc_cb_param_t *evtParam); +#endif private: friend class BLERemoteCharacteristic; - BLERemoteDescriptor(uint16_t handle, BLEUUID uuid, BLERemoteCharacteristic *pRemoteCharacteristic); - uint16_t m_handle; // Server handle of this descriptor. - BLEUUID m_uuid; // UUID of this descriptor. - String m_value; // Last received value of the descriptor. + + /*************************************************************************** + * Common private properties * + ***************************************************************************/ + uint16_t m_handle; // Server handle of this descriptor. + BLEUUID m_uuid; // UUID of this descriptor. + String m_value; // Last received value of the descriptor. + uint8_t m_auth; BLERemoteCharacteristic *m_pRemoteCharacteristic; // Reference to the Remote characteristic of which this descriptor is associated. FreeRTOS::Semaphore m_semaphoreReadDescrEvt = FreeRTOS::Semaphore("ReadDescrEvt"); FreeRTOS::Semaphore m_semaphoreWriteDescrEvt = FreeRTOS::Semaphore("WriteDescrEvt"); - esp_gatt_auth_req_t m_auth; + + /*************************************************************************** + * Common private declarations * + ***************************************************************************/ + + BLERemoteDescriptor(uint16_t handle, BLEUUID uuid, BLERemoteCharacteristic *pRemoteCharacteristic); + + /*************************************************************************** + * NimBLE private declarations * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) + BLERemoteDescriptor(BLERemoteCharacteristic *pRemoteCharacteristic, const struct ble_gatt_dsc *dsc); + static int onWriteCB(uint16_t conn_handle, const struct ble_gatt_error *error, struct ble_gatt_attr *attr, void *arg); + static int onReadCB(uint16_t conn_handle, const struct ble_gatt_error *error, struct ble_gatt_attr *attr, void *arg); +#endif }; -#endif /* CONFIG_BLUEDROID_ENABLED */ +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ #endif /* COMPONENTS_CPP_UTILS_BLEREMOTEDESCRIPTOR_H_ */ diff --git a/libraries/BLE/src/BLERemoteService.cpp b/libraries/BLE/src/BLERemoteService.cpp index e4cc31dbb33..7baf6908d40 100644 --- a/libraries/BLE/src/BLERemoteService.cpp +++ b/libraries/BLE/src/BLERemoteService.cpp @@ -3,12 +3,21 @@ * * Created on: Jul 8, 2017 * Author: kolban + * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on kolban's and h2zero's work) + * Description: Added support for NimBLE */ + #include "soc/soc_caps.h" #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) + +/*************************************************************************** + * Common includes * + ***************************************************************************/ #include #include "BLERemoteService.h" @@ -17,97 +26,14 @@ #include #include "esp32-hal-log.h" -#pragma GCC diagnostic warning "-Wunused-but-set-parameter" - -BLERemoteService::BLERemoteService(esp_gatt_id_t srvcId, BLEClient *pClient, uint16_t startHandle, uint16_t endHandle) { - - log_v(">> BLERemoteService()"); - m_srvcId = srvcId; - m_pClient = pClient; - m_uuid = BLEUUID(m_srvcId); - m_haveCharacteristics = false; - m_startHandle = startHandle; - m_endHandle = endHandle; - - log_v("<< BLERemoteService()"); -} +/*************************************************************************** + * Common functions * + ***************************************************************************/ BLERemoteService::~BLERemoteService() { removeCharacteristics(); } -/* -static bool compareSrvcId(esp_gatt_srvc_id_t id1, esp_gatt_srvc_id_t id2) { - if (id1.id.inst_id != id2.id.inst_id) { - return false; - } - if (!BLEUUID(id1.id.uuid).equals(BLEUUID(id2.id.uuid))) { - return false; - } - return true; -} // compareSrvcId -*/ - -/** - * @brief Handle GATT Client events - */ -void BLERemoteService::gattClientEventHandler(esp_gattc_cb_event_t event, esp_gatt_if_t gattc_if, esp_ble_gattc_cb_param_t *evtParam) { - switch (event) { - // - // ESP_GATTC_GET_CHAR_EVT - // - // get_char: - // - esp_gatt_status_t status - // - uin1t6_t conn_id - // - esp_gatt_srvc_id_t srvc_id - // - esp_gatt_id_t char_id - // - esp_gatt_char_prop_t char_prop - // - /* - case ESP_GATTC_GET_CHAR_EVT: { - // Is this event for this service? If yes, then the local srvc_id and the event srvc_id will be - // the same. - if (compareSrvcId(m_srvcId, evtParam->get_char.srvc_id) == false) { - break; - } - - // If the status is NOT OK then we have a problem and continue. - if (evtParam->get_char.status != ESP_GATT_OK) { - m_semaphoreGetCharEvt.give(); - break; - } - - // This is an indication that we now have the characteristic details for a characteristic owned - // by this service so remember it. - m_characteristicMap.insert(std::pair( - BLEUUID(evtParam->get_char.char_id.uuid).toString().c_str(), - new BLERemoteCharacteristic(evtParam->get_char.char_id, evtParam->get_char.char_prop, this) )); - - - // Now that we have received a characteristic, lets ask for the next one. - esp_err_t errRc = ::esp_ble_gattc_get_characteristic( - m_pClient->getGattcIf(), - m_pClient->getConnId(), - &m_srvcId, - &evtParam->get_char.char_id); - if (errRc != ESP_OK) { - log_e("esp_ble_gattc_get_characteristic: rc=%d %s", errRc, GeneralUtils::errorToString(errRc)); - break; - } - - //m_semaphoreGetCharEvt.give(); - break; - } // ESP_GATTC_GET_CHAR_EVT -*/ - default: break; - } // switch - - // Send the event to each of the characteristics owned by this service. - for (auto &myPair : m_characteristicMapByHandle) { - myPair.second->gattClientEventHandler(event, gattc_if, evtParam); - } -} // gattClientEventHandler - /** * @brief Get the remote characteristic object for the characteristic UUID. * @param [in] uuid Remote characteristic uuid. @@ -144,52 +70,6 @@ BLERemoteCharacteristic *BLERemoteService::getCharacteristic(BLEUUID uuid) { return nullptr; } // getCharacteristic -/** - * @brief Retrieve all the characteristics for this service. - * This function will not return until we have all the characteristics. - * @return N/A - */ -void BLERemoteService::retrieveCharacteristics() { - log_v(">> getCharacteristics() for service: %s", getUUID().toString().c_str()); - - removeCharacteristics(); // Forget any previous characteristics. - - uint16_t offset = 0; - esp_gattc_char_elem_t result; - while (true) { - uint16_t count = 1; // only room for 1 result allocated, so go one by one - esp_gatt_status_t status = - ::esp_ble_gattc_get_all_char(getClient()->getGattcIf(), getClient()->getConnId(), m_startHandle, m_endHandle, &result, &count, offset); - - if (status == ESP_GATT_INVALID_OFFSET) { // We have reached the end of the entries. - break; - } - - if (status != ESP_GATT_OK) { // If we got an error, end. - log_e("esp_ble_gattc_get_all_char: %s", BLEUtils::gattStatusToString(status).c_str()); - break; - } - - if (count == 0) { // If we failed to get any new records, end. - break; - } - - log_d("Found a characteristic: Handle: %d, UUID: %s", result.char_handle, BLEUUID(result.uuid).toString().c_str()); - - // We now have a new characteristic ... let us add that to our set of known characteristics - BLERemoteCharacteristic *pNewRemoteCharacteristic = new BLERemoteCharacteristic(result.char_handle, BLEUUID(result.uuid), result.properties, this); - - m_characteristicMap.insert( - std::pair(pNewRemoteCharacteristic->getUUID().toString().c_str(), pNewRemoteCharacteristic) - ); - m_characteristicMapByHandle.insert(std::pair(result.char_handle, pNewRemoteCharacteristic)); - offset++; // Increment our count of number of descriptors found. - } // Loop forever (until we break inside the loop). - - m_haveCharacteristics = true; // Remember that we have received the characteristics. - log_v("<< getCharacteristics()"); -} // getCharacteristics - /** * @brief Retrieve a map of all the characteristics of this service. * @return A map of all the characteristics of this service. @@ -248,10 +128,6 @@ uint16_t BLERemoteService::getEndHandle() { return m_endHandle; } // getEndHandle -esp_gatt_id_t *BLERemoteService::getSrvcId() { - return &m_srvcId; -} // getSrvcId - uint16_t BLERemoteService::getStartHandle() { return m_startHandle; } // getStartHandle @@ -330,5 +206,222 @@ String BLERemoteService::toString() { return res; } // toString -#endif /* CONFIG_BLUEDROID_ENABLED */ +/*************************************************************************** + * Bluedroid functions * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) + +BLERemoteService::BLERemoteService(esp_gatt_id_t srvcId, BLEClient *pClient, uint16_t startHandle, uint16_t endHandle) { + log_v(">> BLERemoteService()"); + m_srvcId = srvcId; + m_pClient = pClient; + m_uuid = BLEUUID(m_srvcId); + m_haveCharacteristics = false; + m_startHandle = startHandle; + m_endHandle = endHandle; + + log_v("<< BLERemoteService()"); +} + +esp_gatt_id_t *BLERemoteService::getSrvcId() { + return &m_srvcId; +} // getSrvcId + +/** + * @brief Retrieve all the characteristics for this service. + * This function will not return until we have all the characteristics. + * @return N/A + */ +void BLERemoteService::retrieveCharacteristics() { + log_v(">> getCharacteristics() for service: %s", getUUID().toString().c_str()); + + removeCharacteristics(); // Forget any previous characteristics. + + uint16_t offset = 0; + esp_gattc_char_elem_t result; + while (true) { + uint16_t count = 1; // only room for 1 result allocated, so go one by one + esp_gatt_status_t status = + ::esp_ble_gattc_get_all_char(getClient()->getGattcIf(), getClient()->getConnId(), m_startHandle, m_endHandle, &result, &count, offset); + + if (status == ESP_GATT_INVALID_OFFSET) { // We have reached the end of the entries. + break; + } + + if (status != ESP_GATT_OK) { // If we got an error, end. + log_e("esp_ble_gattc_get_all_char: %s", BLEUtils::gattStatusToString(status).c_str()); + break; + } + + if (count == 0) { // If we failed to get any new records, end. + break; + } + + log_d("Found a characteristic: Handle: %d, UUID: %s", result.char_handle, BLEUUID(result.uuid).toString().c_str()); + + // We now have a new characteristic ... let us add that to our set of known characteristics + BLERemoteCharacteristic *pNewRemoteCharacteristic = new BLERemoteCharacteristic(result.char_handle, BLEUUID(result.uuid), result.properties, this); + + m_characteristicMap.insert( + std::pair(pNewRemoteCharacteristic->getUUID().toString().c_str(), pNewRemoteCharacteristic) + ); + m_characteristicMapByHandle.insert(std::pair(result.char_handle, pNewRemoteCharacteristic)); + offset++; // Increment our count of number of descriptors found. + } // Loop forever (until we break inside the loop). + + m_haveCharacteristics = true; // Remember that we have received the characteristics. + log_v("<< getCharacteristics()"); +} // getCharacteristics + +/** + * @brief Handle GATT Client events + */ +void BLERemoteService::gattClientEventHandler(esp_gattc_cb_event_t event, esp_gatt_if_t gattc_if, esp_ble_gattc_cb_param_t *evtParam) { + switch (event) { + // + // ESP_GATTC_GET_CHAR_EVT + // + // get_char: + // - esp_gatt_status_t status + // - uin1t6_t conn_id + // - esp_gatt_srvc_id_t srvc_id + // - esp_gatt_id_t char_id + // - esp_gatt_char_prop_t char_prop + // + /* + case ESP_GATTC_GET_CHAR_EVT: { + // Is this event for this service? If yes, then the local srvc_id and the event srvc_id will be + // the same. + if (compareSrvcId(m_srvcId, evtParam->get_char.srvc_id) == false) { + break; + } + + // If the status is NOT OK then we have a problem and continue. + if (evtParam->get_char.status != ESP_GATT_OK) { + m_semaphoreGetCharEvt.give(); + break; + } + + // This is an indication that we now have the characteristic details for a characteristic owned + // by this service so remember it. + m_characteristicMap.insert(std::pair( + BLEUUID(evtParam->get_char.char_id.uuid).toString().c_str(), + new BLERemoteCharacteristic(evtParam->get_char.char_id, evtParam->get_char.char_prop, this) )); + + + // Now that we have received a characteristic, lets ask for the next one. + esp_err_t errRc = ::esp_ble_gattc_get_characteristic( + m_pClient->getGattcIf(), + m_pClient->getConnId(), + &m_srvcId, + &evtParam->get_char.char_id); + if (errRc != ESP_OK) { + log_e("esp_ble_gattc_get_characteristic: rc=%d %s", errRc, GeneralUtils::errorToString(errRc)); + break; + } + + //m_semaphoreGetCharEvt.give(); + break; + } // ESP_GATTC_GET_CHAR_EVT +*/ + default: break; + } // switch + + // Send the event to each of the characteristics owned by this service. + for (auto &myPair : m_characteristicMapByHandle) { + myPair.second->gattClientEventHandler(event, gattc_if, evtParam); + } +} // gattClientEventHandler + +#endif + +/*************************************************************************** + * NimBLE functions * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) + +BLERemoteService::BLERemoteService(BLEClient *pClient, const struct ble_gatt_svc *service) { + log_v(">> BLERemoteService()"); + m_pClient = pClient; + switch (service->uuid.u.type) { + case BLE_UUID_TYPE_16: m_uuid = BLEUUID(service->uuid.u16.value); break; + case BLE_UUID_TYPE_32: m_uuid = BLEUUID(service->uuid.u32.value); break; + case BLE_UUID_TYPE_128: m_uuid = BLEUUID(const_cast(&service->uuid.u128)); break; + default: break; + } + m_startHandle = service->start_handle; + m_endHandle = service->end_handle; + m_haveCharacteristics = false; + log_v("<< BLERemoteService(): %s", m_uuid.toString().c_str()); +} + +/** + * @brief Retrieve all the characteristics for this service. + * This function will not return until we have all the characteristics. + */ +void BLERemoteService::retrieveCharacteristics() { + log_v(">> retrieveCharacteristics() for service: %s", getUUID().toString().c_str()); + + int rc = 0; + BLETaskData taskData(const_cast(this)); + + rc = ble_gattc_disc_all_chrs(m_pClient->getConnId(), m_startHandle, m_endHandle, BLERemoteService::characteristicDiscCB, &taskData); + + if (rc != 0) { + log_e("ble_gattc_disc_all_chrs: rc=%d %s", rc, BLEUtils::returnCodeToString(rc)); + return; + } + + BLEUtils::taskWait(taskData, BLE_NPL_TIME_FOREVER); + rc = taskData.m_flags; + if (rc == 0 || rc == BLE_HS_EDONE) { + log_d("<< retrieveCharacteristics()"); + return; + } + + log_e("<< retrieveCharacteristics() rc=%d %s", rc, BLEUtils::returnCodeToString(rc)); +} // retrieveCharacteristics + +/** + * @brief Callback for characteristic discovery. + * @return success == 0 or error code. + */ +int BLERemoteService::characteristicDiscCB(uint16_t conn_handle, const struct ble_gatt_error *error, const struct ble_gatt_chr *chr, void *arg) { + log_d("Characteristic Discovered >> status: %d handle: %d", error->status, (error->status == 0) ? chr->val_handle : -1); + + BLETaskData *pTaskData = (BLETaskData *)arg; + BLERemoteService *service = (BLERemoteService *)pTaskData->m_pInstance; + + if (error->status == BLE_HS_ENOTCONN) { + log_e("<< Characteristic Discovery; Not connected"); + BLEUtils::taskRelease(*pTaskData, error->status); + return error->status; + } + + // Make sure the discovery is for this device + if (service->getClient()->getConnId() != conn_handle) { + return 0; + } + + if (error->status == 0) { + // Found a service - add it to the vector + BLERemoteCharacteristic *pRemoteCharacteristic = new BLERemoteCharacteristic(service, chr); + service->m_characteristicMap.insert( + std::pair(pRemoteCharacteristic->getUUID().toString().c_str(), pRemoteCharacteristic) + ); + service->m_characteristicMapByHandle.insert(std::pair(chr->val_handle, pRemoteCharacteristic)); + return 0; + } + + BLEUtils::taskRelease(*pTaskData, error->status); + service->m_haveCharacteristics = true; + log_d("<< Characteristic Discovered"); + return error->status; +} + +#endif + +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ diff --git a/libraries/BLE/src/BLERemoteService.h b/libraries/BLE/src/BLERemoteService.h index 49845a0a1e8..c93d91f6852 100644 --- a/libraries/BLE/src/BLERemoteService.h +++ b/libraries/BLE/src/BLERemoteService.h @@ -3,6 +3,10 @@ * * Created on: Jul 8, 2017 * Author: kolban + * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on kolban's and h2zero's work) + * Description: Added support for NimBLE */ #ifndef COMPONENTS_CPP_UTILS_BLEREMOTESERVICE_H_ @@ -11,7 +15,11 @@ #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) + +/*************************************************************************** + * Common includes * + ***************************************************************************/ #include @@ -19,6 +27,19 @@ #include "BLERemoteCharacteristic.h" #include "BLEUUID.h" #include "RTOS.h" +#include "BLEUtils.h" + +/*************************************************************************** + * NimBLE includes * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) +#include +#endif + +/*************************************************************************** + * Forward declarations * + ***************************************************************************/ class BLEClient; class BLERemoteCharacteristic; @@ -28,15 +49,18 @@ class BLERemoteCharacteristic; */ class BLERemoteService { public: - virtual ~BLERemoteService(); + /*************************************************************************** + * Common public declarations * + ***************************************************************************/ - // Public methods + virtual ~BLERemoteService(); BLERemoteCharacteristic *getCharacteristic(const char *uuid); // Get the specified characteristic reference. BLERemoteCharacteristic *getCharacteristic(BLEUUID uuid); // Get the specified characteristic reference. BLERemoteCharacteristic *getCharacteristic(uint16_t uuid); // Get the specified characteristic reference. std::map *getCharacteristics(); std::map *getCharacteristicsByHandle(); // Get the characteristics map. void getCharacteristics(std::map **pCharacteristicMap); + void retrieveCharacteristics(); BLEClient *getClient(void); // Get a reference to the client associated with this service. uint16_t getHandle(); // Get the handle of this service. @@ -46,24 +70,12 @@ class BLERemoteService { String toString(void); private: - // Private constructor ... never meant to be created by a user application. - BLERemoteService(esp_gatt_id_t srvcId, BLEClient *pClient, uint16_t startHandle, uint16_t endHandle); - - // Friends friend class BLEClient; friend class BLERemoteCharacteristic; - // Private methods - void retrieveCharacteristics(void); // Retrieve the characteristics from the BLE Server. - esp_gatt_id_t *getSrvcId(void); - uint16_t getStartHandle(); // Get the start handle for this service. - uint16_t getEndHandle(); // Get the end handle for this service. - - void gattClientEventHandler(esp_gattc_cb_event_t event, esp_gatt_if_t gattc_if, esp_ble_gattc_cb_param_t *evtParam); - - void removeCharacteristics(); - - // Properties + /*************************************************************************** + * Common private properties * + ***************************************************************************/ // We maintain a map of characteristics owned by this service keyed by a string representation of the UUID. std::map m_characteristicMap; @@ -74,12 +86,47 @@ class BLERemoteService { bool m_haveCharacteristics; // Have we previously obtained the characteristics. BLEClient *m_pClient; FreeRTOS::Semaphore m_semaphoreGetCharEvt = FreeRTOS::Semaphore("GetCharEvt"); - esp_gatt_id_t m_srvcId; BLEUUID m_uuid; // The UUID of this service. uint16_t m_startHandle; // The starting handle of this service. uint16_t m_endHandle; // The ending handle of this service. + + /*************************************************************************** + * Bluedroid private properties * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) + esp_gatt_id_t m_srvcId; +#endif + + /*************************************************************************** + * Common private declarations * + ***************************************************************************/ + + uint16_t getStartHandle(); // Get the start handle for this service. + uint16_t getEndHandle(); // Get the end handle for this service. + void removeCharacteristics(); + + /*************************************************************************** + * Bluedroid private declarations * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) + // Private constructor ... never meant to be created by a user application. + BLERemoteService(esp_gatt_id_t srvcId, BLEClient *pClient, uint16_t startHandle, uint16_t endHandle); + esp_gatt_id_t *getSrvcId(); + void gattClientEventHandler(esp_gattc_cb_event_t event, esp_gatt_if_t gattc_if, esp_ble_gattc_cb_param_t *evtParam); +#endif + + /*************************************************************************** + * NimBLE private declarations * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) + BLERemoteService(BLEClient *pClient, const struct ble_gatt_svc *service); + static int characteristicDiscCB(uint16_t conn_handle, const struct ble_gatt_error *error, const struct ble_gatt_chr *chr, void *arg); +#endif }; // BLERemoteService -#endif /* CONFIG_BLUEDROID_ENABLED */ +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ #endif /* COMPONENTS_CPP_UTILS_BLEREMOTESERVICE_H_ */ diff --git a/libraries/BLE/src/BLEScan.cpp b/libraries/BLE/src/BLEScan.cpp index 0a99b46c61d..6d18bb2d751 100644 --- a/libraries/BLE/src/BLEScan.cpp +++ b/libraries/BLE/src/BLEScan.cpp @@ -6,12 +6,21 @@ * * Update: April, 2021 * add BLE5 support + * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on kolban's and h2zero's work) + * Description: Added support for NimBLE */ + #include "soc/soc_caps.h" #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) + +/*************************************************************************** + * Common includes * + ***************************************************************************/ #include @@ -23,12 +32,17 @@ #include "GeneralUtils.h" #include "esp32-hal-log.h" +/*************************************************************************** + * Common functions * + ***************************************************************************/ + /** - * Constructor + * @brief Constructor */ BLEScan::BLEScan() { memset(&m_scan_params, 0, sizeof(m_scan_params)); // Initialize all params - m_scan_params.scan_type = BLE_SCAN_TYPE_PASSIVE; // Default is a passive scan. +#if defined(CONFIG_BLUEDROID_ENABLED) + m_scan_params.scan_type = BLE_SCAN_TYPE_PASSIVE; // Default is a passive scan. m_scan_params.own_addr_type = BLE_ADDR_TYPE_PUBLIC; m_scan_params.scan_filter_policy = BLE_SCAN_FILTER_ALLOW_ALL; m_scan_params.scan_duplicate = BLE_SCAN_DUPLICATE_DISABLE; @@ -38,15 +52,246 @@ BLEScan::BLEScan() { m_shouldParse = true; setInterval(100); setWindow(100); +#endif + +#if defined(CONFIG_NIMBLE_ENABLED) + m_scan_params.filter_policy = BLE_HCI_SCAN_FILT_NO_WL; + m_scan_params.passive = 1; // If set, don’t send scan requests to advertisers (i.e., don’t request additional advertising data). + m_scan_params.limited = 0; // If set, only discover devices in limited discoverable mode. + m_scan_params.filter_duplicates = 1; // If set, the controller ignores all but the first advertisement from each device. + m_pAdvertisedDeviceCallbacks = nullptr; + m_ignoreResults = false; + m_pTaskData = nullptr; + m_duration = BLE_HS_FOREVER; // make sure this is non-zero in the event of a host reset + m_maxResults = 0xFF; + m_stopped = true; + m_wantDuplicates = false; + m_shouldParse = true; + // This is defined as the time interval from when the Controller started its last LE scan until it begins the subsequent LE scan. (units=0.625 msec) + setInterval(100); + // The duration of the LE scan. LE_Scan_Window shall be less than or equal to LE_Scan_Interval (units=0.625 msec) + setWindow(100); +#endif } // BLEScan +/** + * @brief Scan destructor, release any allocated resources. + */ +BLEScan::~BLEScan() { + clearResults(); +} + +/** + * @brief Should we perform an active or passive scan? + * The default is a passive scan. An active scan means that we will wish a scan response. + * @param [in] active If true, we perform an active scan otherwise a passive scan. + * @return N/A. + */ +void BLEScan::setActiveScan(bool active) { +#if defined(CONFIG_BLUEDROID_ENABLED) + if (active) { + m_scan_params.scan_type = BLE_SCAN_TYPE_ACTIVE; + } else { + m_scan_params.scan_type = BLE_SCAN_TYPE_PASSIVE; + } +#endif + +#if defined(CONFIG_NIMBLE_ENABLED) + m_scan_params.passive = !active; +#endif +} // setActiveScan + +/** + * @brief Set the call backs to be invoked. + * @param [in] pAdvertisedDeviceCallbacks Call backs to be invoked. + * @param [in] wantDuplicates True if we wish to be called back with duplicates. Default is false. + * @param [in] shouldParse True if we wish to parse advertised package or raw payload. Default is true. + */ +void BLEScan::setAdvertisedDeviceCallbacks(BLEAdvertisedDeviceCallbacks *pAdvertisedDeviceCallbacks, bool wantDuplicates, bool shouldParse) { + m_wantDuplicates = wantDuplicates; +#if defined(CONFIG_NIMBLE_ENABLED) + setDuplicateFilter(!wantDuplicates); +#endif + m_pAdvertisedDeviceCallbacks = pAdvertisedDeviceCallbacks; + m_shouldParse = shouldParse; +} // setAdvertisedDeviceCallbacks + +/** + * @brief Set the interval to scan. + * @param [in] The interval in msecs. + */ +void BLEScan::setInterval(uint16_t intervalMSecs) { +#if defined(CONFIG_BLUEDROID_ENABLED) + m_scan_params.scan_interval = intervalMSecs / 0.625; +#endif + +#if defined(CONFIG_NIMBLE_ENABLED) + m_scan_params.itvl = intervalMSecs / 0.625; +#endif +} // setInterval + +/** + * @brief Set the window to actively scan. + * @param [in] windowMSecs How long to actively scan. + */ +void BLEScan::setWindow(uint16_t windowMSecs) { +#if defined(CONFIG_BLUEDROID_ENABLED) + m_scan_params.scan_window = windowMSecs / 0.625; +#endif + +#if defined(CONFIG_NIMBLE_ENABLED) + m_scan_params.window = windowMSecs / 0.625; +#endif +} // setWindow + +// delete peer device from cache after disconnecting, it is required in case we are connecting to devices with not public address +void BLEScan::erase(BLEAddress address) { + log_i("erase device: %s", address.toString().c_str()); + BLEAdvertisedDevice *advertisedDevice = m_scanResults.m_vectorAdvertisedDevices.find(address.toString().c_str())->second; + m_scanResults.m_vectorAdvertisedDevices.erase(address.toString().c_str()); + delete advertisedDevice; +} + +/** + * @brief Dump the scan results to the log. + */ +void BLEScanResults::dump() { + log_v(">> Dump scan results:"); + for (int i = 0; i < getCount(); i++) { + log_d("- %s", getDevice(i).toString().c_str()); + } +} // dump + +/** + * @brief Return the count of devices found in the last scan. + * @return The number of devices found in the last scan. + */ +int BLEScanResults::getCount() { + return m_vectorAdvertisedDevices.size(); +} // getCount + +/** + * @brief Return the specified device at the given index. + * The index should be between 0 and getCount()-1. + * @param [in] i The index of the device. + * @return The device at the specified index. + */ +BLEAdvertisedDevice BLEScanResults::getDevice(uint32_t i) { + uint32_t x = 0; + BLEAdvertisedDevice dev = *m_vectorAdvertisedDevices.begin()->second; + for (auto it = m_vectorAdvertisedDevices.begin(); it != m_vectorAdvertisedDevices.end(); it++) { + dev = *it->second; + if (x == i) { + break; + } + x++; + } + return dev; +} + +BLEScanResults *BLEScan::getResults() { + return &m_scanResults; +} + +void BLEScan::clearResults() { + for (auto _dev : m_scanResults.m_vectorAdvertisedDevices) { + delete _dev.second; + } + m_scanResults.m_vectorAdvertisedDevices.clear(); +} + +/*************************************************************************** + * Bluedroid functions * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) + +#if defined(SOC_BLE_50_SUPPORTED) + +void BLEScan::setExtendedScanCallback(BLEExtAdvertisingCallbacks *cb) { + m_pExtendedScanCb = cb; +} + +/** +* @brief This function is used to enable scanning. +* +* @param[in] duration : Scan duration +* @param[in] period : Time interval from when the Controller started its last Scan Duration until it begins the subsequent Scan Duration. +* +* @return - ESP_OK : success +* - other : failed +* +*/ +esp_err_t BLEScan::startExtScan(uint32_t duration, uint16_t period) { + esp_err_t rc = esp_ble_gap_start_ext_scan(duration, period); + if (rc) { + log_e("extended scan start failed: %d", rc); + } + return rc; +} + +esp_err_t BLEScan::stopExtScan() { + esp_err_t rc = esp_ble_gap_stop_ext_scan(); + if (rc) { + log_e("extended scan stop failed: %d", rc); + } + return rc; +} + +void BLEScan::setPeriodicScanCallback(BLEPeriodicScanCallbacks *cb) { + m_pPeriodicScanCb = cb; +} + +/** +* @brief This function is used to set the extended scan parameters to be used on the advertising channels. +* +* +* @return - ESP_OK : success +* - other : failed +* +*/ +esp_err_t BLEScan::setExtScanParams() { + esp_ble_ext_scan_params_t ext_scan_params = { + .own_addr_type = BLE_ADDR_TYPE_PUBLIC, + .filter_policy = BLE_SCAN_FILTER_ALLOW_ALL, + .scan_duplicate = BLE_SCAN_DUPLICATE_DISABLE, + .cfg_mask = ESP_BLE_GAP_EXT_SCAN_CFG_UNCODE_MASK | ESP_BLE_GAP_EXT_SCAN_CFG_CODE_MASK, + .uncoded_cfg = {BLE_SCAN_TYPE_ACTIVE, 40, 40}, + .coded_cfg = {BLE_SCAN_TYPE_ACTIVE, 40, 40}, + }; + + esp_err_t rc = esp_ble_gap_set_ext_scan_params(&ext_scan_params); + if (rc) { + log_e("set extend scan params error, error code = %x", rc); + } + return rc; +} + +/** +* @brief This function is used to set the extended scan parameters to be used on the advertising channels. +* +* @param[in] params : scan parameters +* +* @return - ESP_OK : success +* - other : failed +* +*/ +esp_err_t BLEScan::setExtScanParams(esp_ble_ext_scan_params_t *ext_scan_params) { + esp_err_t rc = esp_ble_gap_set_ext_scan_params(ext_scan_params); + if (rc) { + log_e("set extend scan params error, error code = %x", rc); + } + return rc; +} + +#endif // SOC_BLE_50_SUPPORTED + /** * @brief Handle GAP events related to scans. * @param [in] event The event type for this event. * @param [in] param Parameter data for this event. */ void BLEScan::handleGAPEvent(esp_gap_ble_cb_event_t event, esp_ble_gap_cb_param_t *param) { - switch (event) { // --------------------------- @@ -118,6 +363,7 @@ void BLEScan::handleGAPEvent(esp_gap_ble_cb_event_t event, esp_ble_gap_cb_param_ advertisedDevice->setAddress(advertisedAddress); advertisedDevice->setRSSI(param->scan_rst.rssi); advertisedDevice->setAdFlag(param->scan_rst.flag); + advertisedDevice->setAdvType(param->scan_rst.ble_evt_type); if (m_shouldParse) { advertisedDevice->parseAdvertisement((uint8_t *)param->scan_rst.ble_adv, param->scan_rst.adv_data_len + param->scan_rst.scan_rsp_len); } else { @@ -245,126 +491,6 @@ void BLEScan::handleGAPEvent(esp_gap_ble_cb_event_t event, esp_ble_gap_cb_param_ } // End switch } // gapEventHandler -/** - * @brief Should we perform an active or passive scan? - * The default is a passive scan. An active scan means that we will wish a scan response. - * @param [in] active If true, we perform an active scan otherwise a passive scan. - * @return N/A. - */ -void BLEScan::setActiveScan(bool active) { - if (active) { - m_scan_params.scan_type = BLE_SCAN_TYPE_ACTIVE; - } else { - m_scan_params.scan_type = BLE_SCAN_TYPE_PASSIVE; - } -} // setActiveScan - -/** - * @brief Set the call backs to be invoked. - * @param [in] pAdvertisedDeviceCallbacks Call backs to be invoked. - * @param [in] wantDuplicates True if we wish to be called back with duplicates. Default is false. - * @param [in] shouldParse True if we wish to parse advertised package or raw payload. Default is true. - */ -void BLEScan::setAdvertisedDeviceCallbacks(BLEAdvertisedDeviceCallbacks *pAdvertisedDeviceCallbacks, bool wantDuplicates, bool shouldParse) { - m_wantDuplicates = wantDuplicates; - m_pAdvertisedDeviceCallbacks = pAdvertisedDeviceCallbacks; - m_shouldParse = shouldParse; -} // setAdvertisedDeviceCallbacks - -#ifdef SOC_BLE_50_SUPPORTED - -void BLEScan::setExtendedScanCallback(BLEExtAdvertisingCallbacks *cb) { - m_pExtendedScanCb = cb; -} - -/** -* @brief This function is used to set the extended scan parameters to be used on the advertising channels. -* -* -* @return - ESP_OK : success -* - other : failed -* -*/ -esp_err_t BLEScan::setExtScanParams() { - esp_ble_ext_scan_params_t ext_scan_params = { - .own_addr_type = BLE_ADDR_TYPE_PUBLIC, - .filter_policy = BLE_SCAN_FILTER_ALLOW_ALL, - .scan_duplicate = BLE_SCAN_DUPLICATE_DISABLE, - .cfg_mask = ESP_BLE_GAP_EXT_SCAN_CFG_UNCODE_MASK | ESP_BLE_GAP_EXT_SCAN_CFG_CODE_MASK, - .uncoded_cfg = {BLE_SCAN_TYPE_ACTIVE, 40, 40}, - .coded_cfg = {BLE_SCAN_TYPE_ACTIVE, 40, 40}, - }; - - esp_err_t rc = esp_ble_gap_set_ext_scan_params(&ext_scan_params); - if (rc) { - log_e("set extend scan params error, error code = %x", rc); - } - return rc; -} - -/** -* @brief This function is used to set the extended scan parameters to be used on the advertising channels. -* -* @param[in] params : scan parameters -* -* @return - ESP_OK : success -* - other : failed -* -*/ -esp_err_t BLEScan::setExtScanParams(esp_ble_ext_scan_params_t *ext_scan_params) { - esp_err_t rc = esp_ble_gap_set_ext_scan_params(ext_scan_params); - if (rc) { - log_e("set extend scan params error, error code = %x", rc); - } - return rc; -} - -/** -* @brief This function is used to enable scanning. -* -* @param[in] duration : Scan duration -* @param[in] period : Time interval from when the Controller started its last Scan Duration until it begins the subsequent Scan Duration. -* -* @return - ESP_OK : success -* - other : failed -* -*/ -esp_err_t BLEScan::startExtScan(uint32_t duration, uint16_t period) { - esp_err_t rc = esp_ble_gap_start_ext_scan(duration, period); - if (rc) { - log_e("extended scan start failed: %d", rc); - } - return rc; -} - -esp_err_t BLEScan::stopExtScan() { - esp_err_t rc; - rc = esp_ble_gap_stop_ext_scan(); - - return rc; -} - -void BLEScan::setPeriodicScanCallback(BLEPeriodicScanCallbacks *cb) { - m_pPeriodicScanCb = cb; -} - -#endif // SOC_BLE_50_SUPPORTED -/** - * @brief Set the interval to scan. - * @param [in] The interval in msecs. - */ -void BLEScan::setInterval(uint16_t intervalMSecs) { - m_scan_params.scan_interval = intervalMSecs / 0.625; -} // setInterval - -/** - * @brief Set the window to actively scan. - * @param [in] windowMSecs How long to actively scan. - */ -void BLEScan::setWindow(uint16_t windowMSecs) { - m_scan_params.scan_window = windowMSecs / 0.625; -} // setWindow - /** * @brief Start scanning. * @param [in] duration The duration in seconds for which to scan. @@ -425,7 +551,7 @@ BLEScanResults *BLEScan::start(uint32_t duration, bool is_continue) { * @brief Stop an in progress scan. * @return N/A. */ -void BLEScan::stop() { +bool BLEScan::stop() { log_v(">> stop()"); esp_err_t errRc = ::esp_ble_gap_stop_scanning(); @@ -435,67 +561,303 @@ void BLEScan::stop() { if (errRc != ESP_OK) { log_e("esp_ble_gap_stop_scanning: err: %d, text: %s", errRc, GeneralUtils::errorToString(errRc)); - return; + return false; } log_v("<< stop()"); + return true; } // stop -// delete peer device from cache after disconnecting, it is required in case we are connecting to devices with not public address -void BLEScan::erase(BLEAddress address) { - log_i("erase device: %s", address.toString().c_str()); - BLEAdvertisedDevice *advertisedDevice = m_scanResults.m_vectorAdvertisedDevices.find(address.toString().c_str())->second; - m_scanResults.m_vectorAdvertisedDevices.erase(address.toString().c_str()); - delete advertisedDevice; +#endif // CONFIG_BLUEDROID_ENABLED + +/*************************************************************************** + * NimBLE functions * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) + +/** + * @brief Set whether or not the BLE controller should only report results + * from devices it has not already seen. + * @param [in] enabled If true, scanned devices will only be reported once. + * @details The controller has a limited buffer and will start reporting + * duplicate devices once the limit is reached. + */ +void BLEScan::setDuplicateFilter(bool enabled) { + m_scan_params.filter_duplicates = enabled; +} // setDuplicateFilter + +/** + * @brief Get the status of the scanner. + * @return true if scanning or scan starting. + */ +bool BLEScan::isScanning() { + return ble_gap_disc_active(); } /** - * @brief Dump the scan results to the log. + * @brief Handle GAP events related to scans. + * @param [in] event The event type for this event. + * @param [in] param Parameter data for this event. */ -void BLEScanResults::dump() { - log_v(">> Dump scan results:"); - for (int i = 0; i < getCount(); i++) { - log_d("- %s", getDevice(i).toString().c_str()); +int BLEScan::handleGAPEvent(ble_gap_event *event, void *arg) { + BLEScan *pScan = (BLEScan *)arg; + + switch (event->type) { + case BLE_GAP_EVENT_DISC: + { + if (pScan->m_ignoreResults) { + log_i("Scan op in progress - ignoring results"); + return 0; + } + + const auto &disc = event->disc; + const auto event_type = disc.event_type; + const bool isLegacyAdv = true; + BLEAddress advertisedAddress(disc.addr); + + BLEClient *client = BLEDevice::getClientByAddress(advertisedAddress); + if (client != nullptr && client->isConnected()) { + log_i("Client %s connected - ignoring event", advertisedAddress.toString().c_str()); + return 0; + } + + BLEAdvertisedDevice *advertisedDevice = nullptr; + + // If we've seen this device before get a pointer to it from the vector + for (auto &it : pScan->m_scanResults.m_vectorAdvertisedDevices) { + if (it.second->getAddress() == advertisedAddress) { + advertisedDevice = it.second; + break; + } + } + + // If we haven't seen this device before; create a new instance and insert it in the vector. + // Otherwise just update the relevant parameters of the already known device. + if (advertisedDevice == nullptr) { + // Check if we have reach the scan results limit, ignore this one if so. + // We still need to store each device when maxResults is 0 to be able to append the scan results + if (pScan->m_maxResults > 0 && pScan->m_maxResults < 0xFF && (pScan->m_scanResults.m_vectorAdvertisedDevices.size() >= pScan->m_maxResults)) { + return 0; + } + + if (isLegacyAdv && event_type == BLE_HCI_ADV_RPT_EVTYPE_SCAN_RSP) { + log_i("Scan response without advertisement: %s", advertisedAddress.toString().c_str()); + } + + advertisedDevice = new BLEAdvertisedDevice(); + advertisedDevice->setAddress(advertisedAddress); + advertisedDevice->setAddressType(event->disc.addr.type); + advertisedDevice->setAdvType(event_type); + pScan->m_scanResults.m_vectorAdvertisedDevices.insert( + std::pair(advertisedAddress.toString().c_str(), advertisedDevice) + ); + log_i("New advertiser: %s", advertisedAddress.toString().c_str()); + } else if (advertisedDevice != nullptr) { + log_i("Updated advertiser: %s", advertisedAddress.toString().c_str()); + } else { + // Scan response from unknown device + return 0; + } + + advertisedDevice->setRSSI(event->disc.rssi); + + if (pScan->m_shouldParse) { + advertisedDevice->parseAdvertisement((uint8_t *)event->disc.data, event->disc.length_data); + } else { + advertisedDevice->setPayload((uint8_t *)event->disc.data, event->disc.length_data, event_type == BLE_HCI_ADV_RPT_EVTYPE_SCAN_RSP); + } + + advertisedDevice->setScan(pScan); + + if (pScan->m_pAdvertisedDeviceCallbacks) { + // If not active scanning or scan response is not available + // report the result to the callback now. + if (pScan->m_scan_params.passive || !isLegacyAdv || !advertisedDevice->isScannable()) { + advertisedDevice->m_callbackSent = true; + pScan->m_pAdvertisedDeviceCallbacks->onResult(*advertisedDevice); + + // Otherwise, wait for the scan response so we can report the complete data. + } else if (isLegacyAdv && event_type == BLE_HCI_ADV_RPT_EVTYPE_SCAN_RSP) { + advertisedDevice->m_callbackSent = true; + pScan->m_pAdvertisedDeviceCallbacks->onResult(*advertisedDevice); + } + } + + // If not storing results and we have invoked the callback, delete the device. + if (pScan->m_maxResults == 0 && advertisedDevice->m_callbackSent) { + pScan->erase(advertisedAddress); + } + + return 0; + } + + case BLE_GAP_EVENT_DISC_COMPLETE: + { + log_d("discovery complete; reason=%d", event->disc_complete.reason); + + // If a device advertised with scan response available and it was not received + // the callback would not have been invoked, so do it here. + if (pScan->m_pAdvertisedDeviceCallbacks) { + for (auto &it : pScan->m_scanResults.m_vectorAdvertisedDevices) { + if (!it.second->m_callbackSent) { + pScan->m_pAdvertisedDeviceCallbacks->onResult(*(it.second)); + } + } + } + + if (pScan->m_maxResults == 0) { + pScan->clearResults(); + } + + if (pScan->m_scanCompleteCB != nullptr) { + pScan->m_scanCompleteCB(pScan->m_scanResults); + } + + if (pScan->m_pTaskData != nullptr) { + BLEUtils::taskRelease(*pScan->m_pTaskData, event->disc_complete.reason); + } + + return 0; + } + + default: return 0; } -} // dump +} // gapEventHandler /** - * @brief Return the count of devices found in the last scan. - * @return The number of devices found in the last scan. + * @brief Start scanning. + * @param [in] duration The duration in seconds for which to scan. + * @param [in] scanCompleteCB A function to be called when scanning has completed. + * @param [in] is_continue Set to true to save previous scan results, false to clear them. + * @return True if scan started or false if there was an error. */ -int BLEScanResults::getCount() { - return m_vectorAdvertisedDevices.size(); -} // getCount +bool BLEScan::start(uint32_t duration, void (*scanCompleteCB)(BLEScanResults), bool is_continue) { + log_d(">> start(duration=%d)", duration); + + if (!is_continue) { + clearResults(); + } + + // Save the callback to be invoked when the scan completes. + m_scanCompleteCB = scanCompleteCB; + // Save the duration in the case that the host is reset so we can reuse it. + m_duration = duration; + + // If 0 duration specified then we assume a continuous scan is desired. + if (duration == 0) { + duration = BLE_HS_FOREVER; + } else { + // convert duration to milliseconds + duration = duration * 1000; + } + + // Set the flag to ignore the results while we are deleting the vector + if (!is_continue) { + m_ignoreResults = true; + } + + int rc = ble_gap_disc(BLEDevice::m_ownAddrType, duration, &m_scan_params, BLEScan::handleGAPEvent, this); + + switch (rc) { + case 0: + case BLE_HS_EALREADY: break; + + case BLE_HS_EBUSY: log_e("Unable to scan - connection in progress."); break; + + case BLE_HS_ETIMEOUT_HCI: + case BLE_HS_EOS: + case BLE_HS_ECONTROLLER: + case BLE_HS_ENOTSYNCED: log_e("Unable to scan - Host Reset"); break; + + default: log_e("Error initiating GAP discovery procedure; rc=%d, %s", rc, BLEUtils::returnCodeToString(rc)); break; + } + + m_ignoreResults = false; + log_d("<< start()"); + + if (rc != 0 && rc != BLE_HS_EALREADY) { + return false; + } + return true; +} // start /** - * @brief Return the specified device at the given index. - * The index should be between 0 and getCount()-1. - * @param [in] i The index of the device. - * @return The device at the specified index. + * @brief Start scanning and block until scanning has been completed. + * @param [in] duration The duration in seconds for which to scan. + * @param [in] is_continue Set to true to save previous scan results, false to clear them. + * @return The BLEScanResults. */ -BLEAdvertisedDevice BLEScanResults::getDevice(uint32_t i) { - uint32_t x = 0; - BLEAdvertisedDevice dev = *m_vectorAdvertisedDevices.begin()->second; - for (auto it = m_vectorAdvertisedDevices.begin(); it != m_vectorAdvertisedDevices.end(); it++) { - dev = *it->second; - if (x == i) { - break; - } - x++; +BLEScanResults *BLEScan::start(uint32_t duration, bool is_continue) { + if (duration == 0) { + log_w("Blocking scan called with duration = forever"); } - return dev; -} -BLEScanResults *BLEScan::getResults() { + if (m_pTaskData != nullptr) { + log_e("Scan already in progress"); + return &m_scanResults; + } + + BLETaskData taskData(this); + m_pTaskData = &taskData; + + if (start(duration, nullptr, is_continue)) { + BLEUtils::taskWait(taskData, BLE_NPL_TIME_FOREVER); + } + + m_pTaskData = nullptr; return &m_scanResults; +} // start + +/** + * @brief Stop an in progress scan. + * @return True if successful. + */ +bool BLEScan::stop() { + log_d(">> stop()"); + + int rc = ble_gap_disc_cancel(); + if (rc != 0 && rc != BLE_HS_EALREADY) { + log_e("Failed to cancel scan; rc=%d", rc); + return false; + } + + if (m_maxResults == 0) { + clearResults(); + } + + if (rc != BLE_HS_EALREADY && m_scanCompleteCB != nullptr) { + m_scanCompleteCB(m_scanResults); + } + + if (m_pTaskData != nullptr) { + BLEUtils::taskRelease(*m_pTaskData); + } + + log_d("<< stop()"); + return true; +} // stop + +/** + * @brief Called when host reset, we set a flag to stop scanning until synced. + */ +void BLEScan::onHostReset() { + m_ignoreResults = true; } -void BLEScan::clearResults() { - for (auto _dev : m_scanResults.m_vectorAdvertisedDevices) { - delete _dev.second; +/** + * @brief If the host reset and re-synced this is called. + * If the application was scanning indefinitely with a callback, restart it. + */ +void BLEScan::onHostSync() { + m_ignoreResults = false; + + if (m_duration == 0 && m_pAdvertisedDeviceCallbacks != nullptr) { + start(m_duration, m_scanCompleteCB); } - m_scanResults.m_vectorAdvertisedDevices.clear(); } -#endif /* CONFIG_BLUEDROID_ENABLED */ +#endif // CONFIG_NIMBLE_ENABLED + +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ diff --git a/libraries/BLE/src/BLEScan.h b/libraries/BLE/src/BLEScan.h index 080e3b803b2..842b7144f06 100644 --- a/libraries/BLE/src/BLEScan.h +++ b/libraries/BLE/src/BLEScan.h @@ -3,6 +3,10 @@ * * Created on: Jul 1, 2017 * Author: kolban + * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on kolban's and h2zero's work) + * Description: Added support for NimBLE */ #ifndef COMPONENTS_CPP_UTILS_BLESCAN_H_ @@ -11,22 +15,51 @@ #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) -#include +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) + +/*************************************************************************** + * Common includes * + ***************************************************************************/ -// #include #include #include "BLEAdvertisedDevice.h" #include "BLEClient.h" +#include "BLEUtils.h" #include "RTOS.h" +/*************************************************************************** + * Bluedroid includes * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) +#include +#endif + +/*************************************************************************** + * NimBLE includes * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) +#include +#endif + +/*************************************************************************** + * Forward declarations * + ***************************************************************************/ + class BLEAdvertisedDevice; class BLEAdvertisedDeviceCallbacks; class BLEExtAdvertisingCallbacks; class BLEClient; class BLEScan; class BLEPeriodicScanCallbacks; +struct BLETaskData; +/*************************************************************************** + * Bluedroid type definitions * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) struct esp_ble_periodic_adv_sync_estab_param_t { uint8_t status; /*!< periodic advertising sync status */ uint16_t sync_handle; /*!< periodic advertising sync handle */ @@ -37,6 +70,7 @@ struct esp_ble_periodic_adv_sync_estab_param_t { uint16_t period_adv_interval; /*!< periodic advertising interval */ uint8_t adv_clk_accuracy; /*!< periodic advertising clock accuracy */ }; +#endif /** * @brief The result of having performed a scan. @@ -47,12 +81,21 @@ struct esp_ble_periodic_adv_sync_estab_param_t { */ class BLEScanResults { public: + /*************************************************************************** + * Common public declarations * + ***************************************************************************/ + void dump(); int getCount(); BLEAdvertisedDevice getDevice(uint32_t i); private: friend BLEScan; + + /*************************************************************************** + * Common private properties * + ***************************************************************************/ + std::map m_vectorAdvertisedDevices; }; @@ -63,37 +106,52 @@ class BLEScanResults { */ class BLEScan { public: + /*************************************************************************** + * Common public declarations * + ***************************************************************************/ + + ~BLEScan(); void setActiveScan(bool active); void setAdvertisedDeviceCallbacks(BLEAdvertisedDeviceCallbacks *pAdvertisedDeviceCallbacks, bool wantDuplicates = false, bool shouldParse = true); void setInterval(uint16_t intervalMSecs); void setWindow(uint16_t windowMSecs); bool start(uint32_t duration, void (*scanCompleteCB)(BLEScanResults), bool is_continue = false); BLEScanResults *start(uint32_t duration, bool is_continue = false); - void stop(); + bool stop(); void erase(BLEAddress address); BLEScanResults *getResults(); void clearResults(); -#ifdef SOC_BLE_50_SUPPORTED + /*************************************************************************** + * Bluedroid public declarations * + ***************************************************************************/ + +#if defined(SOC_BLE_50_SUPPORTED) && defined(CONFIG_BLUEDROID_ENABLED) void setExtendedScanCallback(BLEExtAdvertisingCallbacks *cb); void setPeriodicScanCallback(BLEPeriodicScanCallbacks *cb); - esp_err_t stopExtScan(); esp_err_t setExtScanParams(); - esp_err_t setExtScanParams(esp_ble_ext_scan_params_t *ext_scan_params); esp_err_t startExtScan(uint32_t duration, uint16_t period); - -private: - BLEExtAdvertisingCallbacks *m_pExtendedScanCb = nullptr; - BLEPeriodicScanCallbacks *m_pPeriodicScanCb = nullptr; + esp_err_t setExtScanParams(esp_ble_ext_scan_params_t *ext_scan_params); #endif // SOC_BLE_50_SUPPORTED + /*************************************************************************** + * NimBLE public declarations * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) + void setDuplicateFilter(bool enabled); + bool isScanning(); + void clearDuplicateCache(); +#endif + private: - BLEScan(); // One doesn't create a new instance instead one asks the BLEDevice for the singleton. friend class BLEDevice; - void handleGAPEvent(esp_gap_ble_cb_event_t event, esp_ble_gap_cb_param_t *param); - esp_ble_scan_params_t m_scan_params; + /*************************************************************************** + * Common private properties * + ***************************************************************************/ + BLEAdvertisedDeviceCallbacks *m_pAdvertisedDeviceCallbacks = nullptr; bool m_stopped = true; bool m_shouldParse = true; @@ -101,21 +159,79 @@ class BLEScan { BLEScanResults m_scanResults; bool m_wantDuplicates; void (*m_scanCompleteCB)(BLEScanResults scanResults); + + /*************************************************************************** + * Bluedroid private properties * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) + esp_ble_scan_params_t m_scan_params; +#if defined(SOC_BLE_50_SUPPORTED) + BLEExtAdvertisingCallbacks *m_pExtendedScanCb = nullptr; + BLEPeriodicScanCallbacks *m_pPeriodicScanCb = nullptr; +#endif // SOC_BLE_50_SUPPORTED +#endif + + /*************************************************************************** + * NimBLE private properties * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) + uint32_t m_duration; + ble_gap_disc_params m_scan_params; + bool m_ignoreResults; + BLETaskData *m_pTaskData; + uint8_t m_maxResults; +#endif + + /*************************************************************************** + * Common private definitions * + ***************************************************************************/ + + BLEScan(); // One doesn't create a new instance instead one asks the BLEDevice for the singleton. + + /*************************************************************************** + * Bluedroid private definitions * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) + void handleGAPEvent(esp_gap_ble_cb_event_t event, esp_ble_gap_cb_param_t *param); +#endif + + /*************************************************************************** + * NimBLE private definitions * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) + void onHostReset(); + void onHostSync(); + static int handleGAPEvent(ble_gap_event *event, void *arg); +#endif }; // BLEScan class BLEPeriodicScanCallbacks { public: + /*************************************************************************** + * Common public declarations * + ***************************************************************************/ + virtual ~BLEPeriodicScanCallbacks() {} + virtual void onLostSync(uint16_t sync_handle) {} + /*************************************************************************** + * Bluedroid public declarations * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) virtual void onCreateSync(esp_bt_status_t status) {} virtual void onCancelSync(esp_bt_status_t status) {} virtual void onTerminateSync(esp_bt_status_t status) {} - virtual void onLostSync(uint16_t sync_handle) {} virtual void onSync(esp_ble_periodic_adv_sync_estab_param_t) {} virtual void onReport(esp_ble_gap_periodic_adv_report_t params) {} virtual void onStop(esp_bt_status_t status) {} +#endif }; -#endif /* CONFIG_BLUEDROID_ENABLED */ +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ #endif /* COMPONENTS_CPP_UTILS_BLESCAN_H_ */ diff --git a/libraries/BLE/src/BLESecurity.cpp b/libraries/BLE/src/BLESecurity.cpp index 34fc3e69e9e..978026a3809 100644 --- a/libraries/BLE/src/BLESecurity.cpp +++ b/libraries/BLE/src/BLESecurity.cpp @@ -3,77 +3,114 @@ * * Created on: Dec 17, 2017 * Author: chegewara + * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on chegewara's and h2zero's work) + * Description: Added support for NimBLE */ #include "soc/soc_caps.h" #if SOC_BLE_SUPPORTED -#include "BLESecurity.h" #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) + +/*************************************************************************** + * Common includes * + ***************************************************************************/ + +#include "BLESecurity.h" +#include "BLEUtils.h" +#include "esp32-hal-log.h" + +/*************************************************************************** + * NimBLE includes * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) +#include +#endif + +/*************************************************************************** + * Common properties * + ***************************************************************************/ + +uint32_t BLESecurity::m_passkey = BLE_SM_DEFAULT_PASSKEY; + +/*************************************************************************** + * Common functions * + ***************************************************************************/ BLESecurity::BLESecurity() {} BLESecurity::~BLESecurity() {} -/* - * @brief Set requested authentication mode - */ -void BLESecurity::setAuthenticationMode(esp_ble_auth_req_t auth_req) { + +void BLESecurity::setAuthenticationMode(uint8_t auth_req) { m_authReq = auth_req; - esp_ble_gap_set_security_param(ESP_BLE_SM_AUTHEN_REQ_MODE, &m_authReq, sizeof(uint8_t)); // <--- setup requested authentication mode +#if defined(CONFIG_BLUEDROID_ENABLED) + esp_ble_gap_set_security_param(ESP_BLE_SM_AUTHEN_REQ_MODE, &m_authReq, sizeof(uint8_t)); +#elif defined(CONFIG_NIMBLE_ENABLED) + BLESecurity::setAuthenticationMode( + (auth_req & BLE_SM_PAIR_AUTHREQ_BOND) != 0, (auth_req & BLE_SM_PAIR_AUTHREQ_MITM) != 0, (auth_req & BLE_SM_PAIR_AUTHREQ_SC) != 0 + ); +#endif } -/** - * @brief Set our device IO capability to let end user perform authorization - * either by displaying or entering generated 6-digits pin code - */ -void BLESecurity::setCapability(esp_ble_io_cap_t iocap) { +void BLESecurity::setCapability(uint8_t iocap) { m_iocap = iocap; +#if defined(CONFIG_BLUEDROID_ENABLED) esp_ble_gap_set_security_param(ESP_BLE_SM_IOCAP_MODE, &iocap, sizeof(uint8_t)); -} // setCapability +#elif defined(CONFIG_NIMBLE_ENABLED) + ble_hs_cfg.sm_io_cap = iocap; +#endif +} -/** - * @brief Init encryption key by server - * @param key_size is value between 7 and 16 - */ void BLESecurity::setInitEncryptionKey(uint8_t init_key) { m_initKey = init_key; +#if defined(CONFIG_BLUEDROID_ENABLED) esp_ble_gap_set_security_param(ESP_BLE_SM_SET_INIT_KEY, &m_initKey, sizeof(uint8_t)); -} // setInitEncryptionKey +#elif defined(CONFIG_NIMBLE_ENABLED) + ble_hs_cfg.sm_our_key_dist = init_key; +#endif +} -/** - * @brief Init encryption key by client - * @param key_size is value between 7 and 16 - */ void BLESecurity::setRespEncryptionKey(uint8_t resp_key) { m_respKey = resp_key; +#if defined(CONFIG_BLUEDROID_ENABLED) esp_ble_gap_set_security_param(ESP_BLE_SM_SET_RSP_KEY, &m_respKey, sizeof(uint8_t)); -} // setRespEncryptionKey +#elif defined(CONFIG_NIMBLE_ENABLED) + ble_hs_cfg.sm_their_key_dist = resp_key; +#endif +} -/** - * - * - */ void BLESecurity::setKeySize(uint8_t key_size) { +#if defined(CONFIG_BLUEDROID_ENABLED) m_keySize = key_size; esp_ble_gap_set_security_param(ESP_BLE_SM_MAX_KEY_SIZE, &m_keySize, sizeof(uint8_t)); -} //setKeySize +#endif +} -/** - * Setup for static PIN connection, call it first and then call setAuthenticationMode eventually to change it - */ void BLESecurity::setStaticPIN(uint32_t pin) { - uint32_t passkey = pin; - esp_ble_gap_set_security_param(ESP_BLE_SM_SET_STATIC_PASSKEY, &passkey, sizeof(uint32_t)); + m_passkey = pin; +#if defined(CONFIG_BLUEDROID_ENABLED) + esp_ble_gap_set_security_param(ESP_BLE_SM_SET_STATIC_PASSKEY, &m_passkey, sizeof(uint32_t)); setCapability(ESP_IO_CAP_OUT); setKeySize(); setAuthenticationMode(ESP_LE_AUTH_REQ_SC_ONLY); setInitEncryptionKey(ESP_BLE_ENC_KEY_MASK | ESP_BLE_ID_KEY_MASK); +#elif defined(CONFIG_NIMBLE_ENABLED) + setCapability(BLE_HS_IO_DISPLAY_ONLY); + setKeySize(); + setAuthenticationMode(false, false, true); // No bonding, no MITM, secure connection only + setInitEncryptionKey(BLE_HS_KEY_DIST_ENC_KEY | BLE_HS_KEY_DIST_ID_KEY); +#endif } -/** - * @brief Debug function to display what keys are exchanged by peers - */ +/*************************************************************************** + * Bluedroid functions * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) char *BLESecurity::esp_key_type_to_str(esp_ble_key_type_t key_type) { char *key_str = nullptr; switch (key_type) { @@ -89,7 +126,34 @@ char *BLESecurity::esp_key_type_to_str(esp_ble_key_type_t key_type) { default: key_str = (char *)"INVALID BLE KEY TYPE"; break; } return key_str; -} // esp_key_type_to_str +} +#endif + +/*************************************************************************** + * NimBLE functions * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) +void BLESecurity::setAuthenticationMode(bool bonding, bool mitm, bool sc) { + m_authReq = bonding ? BLE_SM_PAIR_AUTHREQ_BOND : 0; + m_authReq |= mitm ? BLE_SM_PAIR_AUTHREQ_MITM : 0; + m_authReq |= sc ? BLE_SM_PAIR_AUTHREQ_SC : 0; + ble_hs_cfg.sm_bonding = bonding; + ble_hs_cfg.sm_mitm = mitm; + ble_hs_cfg.sm_sc = sc; +} + +bool BLESecurity::startSecurity(uint16_t connHandle, int *rcPtr) { + int rc = ble_gap_security_initiate(connHandle); + if (rc != 0) { + log_e("ble_gap_security_initiate: rc=%d %s", rc, BLEUtils::returnCodeToString(rc)); + } + if (rcPtr) { + *rcPtr = rc; + } + return rc == 0 || rc == BLE_HS_EALREADY; +} +#endif -#endif /* CONFIG_BLUEDROID_ENABLED */ +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ diff --git a/libraries/BLE/src/BLESecurity.h b/libraries/BLE/src/BLESecurity.h index 2e3a44b42d1..574110e6118 100644 --- a/libraries/BLE/src/BLESecurity.h +++ b/libraries/BLE/src/BLESecurity.h @@ -3,74 +3,148 @@ * * Created on: Dec 17, 2017 * Author: chegewara + * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on chegewara's and h2zero's work) + * Description: Added support for NimBLE */ #ifndef COMPONENTS_CPP_UTILS_BLESECURITY_H_ #define COMPONENTS_CPP_UTILS_BLESECURITY_H_ + #include "soc/soc_caps.h" #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) + +/*************************************************************************** + * Common includes * + ***************************************************************************/ + +#include "WString.h" +/*************************************************************************** + * Bluedroid includes * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) #include +#endif + +/*************************************************************************** + * NimBLE includes * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) +#include +#endif +/*************************************************************************** + * Common definitions * + ***************************************************************************/ + +#define BLE_SM_DEFAULT_PASSKEY 123456 + +/*************************************************************************** + * Forward declarations * + ***************************************************************************/ + +class BLEServer; +class BLEClient; + +/** + * @brief Security management class + */ class BLESecurity { public: + /*************************************************************************** + * Common public declarations * + ***************************************************************************/ + BLESecurity(); virtual ~BLESecurity(); - void setAuthenticationMode(esp_ble_auth_req_t auth_req); - void setCapability(esp_ble_io_cap_t iocap); - void setInitEncryptionKey(uint8_t init_key); - void setRespEncryptionKey(uint8_t resp_key); - void setKeySize(uint8_t key_size = 16); - void setStaticPIN(uint32_t pin); + static void setAuthenticationMode(uint8_t auth_req); + static void setCapability(uint8_t iocap); + static void setInitEncryptionKey(uint8_t init_key); + static void setRespEncryptionKey(uint8_t resp_key); + static void setKeySize(uint8_t key_size = 16); + static void setStaticPIN(uint32_t pin); + + /*************************************************************************** + * Bluedroid public declarations * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) static char *esp_key_type_to_str(esp_ble_key_type_t key_type); +#endif + + /*************************************************************************** + * NimBLE public declarations * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) + static void setAuthenticationMode(bool bonding, bool mitm, bool sc); + static bool startSecurity(uint16_t connHandle, int *rcPtr = nullptr); +#endif private: - esp_ble_auth_req_t m_authReq; - esp_ble_io_cap_t m_iocap; - uint8_t m_initKey; - uint8_t m_respKey; - uint8_t m_keySize; + friend class BLEServer; + friend class BLEClient; + + /*************************************************************************** + * Common private properties * + ***************************************************************************/ + + static uint8_t m_iocap; + static uint8_t m_authReq; + static uint8_t m_initKey; + static uint8_t m_respKey; + static uint32_t m_passkey; + + /*************************************************************************** + * Bluedroid private properties * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) + static uint8_t m_keySize; +#endif }; // BLESecurity -/* +/** * @brief Callbacks to handle GAP events related to authorization */ class BLESecurityCallbacks { public: - virtual ~BLESecurityCallbacks(){}; + /*************************************************************************** + * Common public declarations * + ***************************************************************************/ - /** - * @brief Its request from peer device to input authentication pin code displayed on peer device. - * It requires that our device is capable to input 6-digits code by end user - * @return Return 6-digits integer value from input device - */ + virtual ~BLESecurityCallbacks(){}; virtual uint32_t onPassKeyRequest() = 0; - - /** - * @brief Provide us 6-digits code to perform authentication. - * It requires that our device is capable to display this code to end user - * @param - */ virtual void onPassKeyNotify(uint32_t pass_key) = 0; + virtual bool onSecurityRequest() = 0; + virtual bool onConfirmPIN(uint32_t pin) = 0; - /** - * @brief Here we can make decision if we want to let negotiate authorization with peer device or not - * return Return true if we accept this peer device request - */ + /*************************************************************************** + * Bluedroid public declarations * + ***************************************************************************/ - virtual bool onSecurityRequest() = 0; - /** - * Provide us information when authentication process is completed - */ +#if defined(CONFIG_BLUEDROID_ENABLED) virtual void onAuthenticationComplete(esp_ble_auth_cmpl_t) = 0; +#endif + + /*************************************************************************** + * NimBLE public declarations * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) + virtual void onAuthenticationComplete(ble_gap_conn_desc *) = 0; +#endif - virtual bool onConfirmPIN(uint32_t pin) = 0; }; // BLESecurityCallbacks -#endif /* CONFIG_BLUEDROID_ENABLED */ +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ #endif // COMPONENTS_CPP_UTILS_BLESECURITY_H_ diff --git a/libraries/BLE/src/BLEServer.cpp b/libraries/BLE/src/BLEServer.cpp index a338cf61451..323237e965e 100644 --- a/libraries/BLE/src/BLEServer.cpp +++ b/libraries/BLE/src/BLEServer.cpp @@ -3,14 +3,23 @@ * * Created on: Apr 16, 2017 * Author: kolban + * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on kolban's and h2zero's work) + * Description: Added support for NimBLE */ + #include "soc/soc_caps.h" #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) + +/*************************************************************************** + * Common includes * + ***************************************************************************/ + #include -#include #include "GeneralUtils.h" #include "BLEDevice.h" #include "BLEServer.h" @@ -21,6 +30,27 @@ #include #include "esp32-hal-log.h" +/*************************************************************************** + * Bluedroid includes * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) +#include +#endif + +/*************************************************************************** + * NimBLE includes * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) +#include +#include +#endif + +/*************************************************************************** + * Common functions * + ***************************************************************************/ + /** * @brief Construct a %BLE Server * @@ -28,8 +58,17 @@ * the BLEDevice class. */ BLEServer::BLEServer() { - m_appId = ESP_GATT_IF_NONE; +#ifdef CONFIG_BLUEDROID_ENABLED m_gatts_if = ESP_GATT_IF_NONE; +#endif + +#if defined(CONFIG_NIMBLE_ENABLED) + memset(m_indWait, BLE_HS_CONN_HANDLE_NONE, sizeof(m_indWait)); + m_svcChanged = false; +#endif + + m_appId = ESP_GATT_IF_NONE; + m_gattsStarted = false; m_connectedCount = 0; m_connId = ESP_GATT_IF_NONE; m_pServerCallbacks = nullptr; @@ -37,7 +76,9 @@ BLEServer::BLEServer() { void BLEServer::createApp(uint16_t appId) { m_appId = appId; +#ifdef CONFIG_BLUEDROID_ENABLED registerApp(appId); +#endif } // createApp /** @@ -64,7 +105,9 @@ BLEService *BLEServer::createService(const char *uuid) { */ BLEService *BLEServer::createService(BLEUUID uuid, uint32_t numHandles, uint8_t inst_id) { log_v(">> createService - %s", uuid.toString().c_str()); +#ifdef CONFIG_BLUEDROID_ENABLED m_semaphoreCreateEvt.take("createService"); +#endif // Check that a service with the supplied UUID does not already exist. if (m_serviceMap.getByUUID(uuid) != nullptr) { @@ -76,7 +119,14 @@ BLEService *BLEServer::createService(BLEUUID uuid, uint32_t numHandles, uint8_t m_serviceMap.setByUUID(uuid, pService); // Save a reference to this service being on this server. pService->executeCreate(this); // Perform the API calls to actually create the service. +#ifdef CONFIG_BLUEDROID_ENABLED m_semaphoreCreateEvt.wait("createService"); +#endif + +#ifdef CONFIG_NIMBLE_ENABLED + m_semaphoreCreateEvt.give(); + serviceChanged(); +#endif log_v("<< createService"); return pService; @@ -121,6 +171,190 @@ uint32_t BLEServer::getConnectedCount() { return m_connectedCount; } // getConnectedCount +void BLEServer::start() { + if (m_gattsStarted) { + return; + } + +#ifdef CONFIG_NIMBLE_ENABLED + int rc = ble_gatts_start(); + if (rc != 0) { + log_e("ble_gatts_start; rc=%d, %s", rc, BLEUtils::returnCodeToString(rc)); + return; + } + +#if ARDUHAL_LOG_LEVEL >= ARDUHAL_LOG_LEVEL_DEBUG + ble_gatts_show_local(); +#endif + + BLEService *svc = m_serviceMap.getFirst(); + while (svc != nullptr) { + if (svc->m_removed == 0) { + rc = ble_gatts_find_svc(&svc->getUUID().getNative()->u, &svc->m_handle); + if (rc != 0) { + abort(); + } + } + + BLECharacteristic *chr = svc->m_characteristicMap.getFirst(); + while (chr != nullptr) { + if ((chr->m_properties & BLE_GATT_CHR_F_INDICATE) || (chr->m_properties & BLE_GATT_CHR_F_NOTIFY)) { + m_notifyChrVec.push_back(chr); + } + chr = svc->m_characteristicMap.getNext(); + } + + svc = m_serviceMap.getNext(); + } + +#endif + + m_gattsStarted = true; +} + +/** + * @brief Set the server callbacks. + * + * As a %BLE server operates, it will generate server level events such as a new client connecting or a previous client + * disconnecting. This function can be called to register a callback handler that will be invoked when these + * events are detected. + * + * @param [in] pCallbacks The callbacks to be invoked. + */ +void BLEServer::setCallbacks(BLEServerCallbacks *pCallbacks) { + m_pServerCallbacks = pCallbacks; +} // setCallbacks + +/* + * Remove service + */ +void BLEServer::removeService(BLEService *service) { +#if defined(CONFIG_BLUEDROID_ENABLED) + service->stop(); + service->executeDelete(); + m_serviceMap.removeService(service); +#endif + +#if defined(CONFIG_NIMBLE_ENABLED) + if (service->m_removed == 0) { + int rc = ble_gatts_svc_set_visibility(service->getHandle(), 0); + if (rc != 0) { + return; + } + service->m_removed = NIMBLE_ATT_REMOVE_DELETE; + serviceChanged(); + m_serviceMap.removeService(service); + BLEDevice::getAdvertising()->removeServiceUUID(service->getUUID()); + } +#endif +} + +/** + * @brief Start advertising. + * + * Start the server advertising its existence. This is a convenience function and is equivalent to + * retrieving the advertising object and invoking start upon it. + */ +void BLEServer::startAdvertising() { + log_v(">> startAdvertising"); + BLEDevice::startAdvertising(); + log_v("<< startAdvertising"); +} // startAdvertising + +/* multi connect support */ +/* TODO do some more tweaks */ +void BLEServer::updatePeerMTU(uint16_t conn_id, uint16_t mtu) { + // set mtu in conn_status_t + const std::map::iterator it = m_connectedServersMap.find(conn_id); + if (it != m_connectedServersMap.end()) { + it->second.mtu = mtu; + std::swap(m_connectedServersMap[conn_id], it->second); + } +} + +std::map BLEServer::getPeerDevices(bool _client) { + return m_connectedServersMap; +} + +uint16_t BLEServer::getPeerMTU(uint16_t conn_id) { +#if defined(CONFIG_BLUEDROID_ENABLED) + return m_connectedServersMap.find(conn_id)->second.mtu; +#endif + +#if defined(CONFIG_NIMBLE_ENABLED) + return ble_att_mtu(conn_id); +#endif +} + +void BLEServer::addPeerDevice(void *peer, bool _client, uint16_t conn_id) { + conn_status_t status = {.peer_device = peer, .connected = true, .mtu = 23}; + + m_connectedServersMap.insert(std::pair(conn_id, status)); +} + +bool BLEServer::removePeerDevice(uint16_t conn_id, bool _client) { + return m_connectedServersMap.erase(conn_id) > 0; +} + +void BLEServerCallbacks::onConnect(BLEServer *pServer) { + log_d("BLEServerCallbacks", ">> onConnect(): Default"); + log_d("BLEServerCallbacks", "Device: %s", BLEDevice::toString().c_str()); + log_d("BLEServerCallbacks", "<< onConnect()"); +} // onConnect + +void BLEServerCallbacks::onDisconnect(BLEServer *pServer) { + log_d("BLEServerCallbacks", ">> onDisconnect(): Default"); + log_d("BLEServerCallbacks", "Device: %s", BLEDevice::toString().c_str()); + log_d("BLEServerCallbacks", "<< onDisconnect()"); +} // onDisconnect + +/*************************************************************************** + * Bluedroid functions * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) + +/** + * Allow to connect GATT server to peer device + * Probably can be used in ANCS for iPhone + */ +bool BLEServer::connect(BLEAddress address) { + esp_bd_addr_t addr; + memcpy(&addr, address.getNative(), 6); + // Perform the open connection request against the target BLE Server. + m_semaphoreOpenEvt.take("connect"); + esp_err_t errRc = ::esp_ble_gatts_open( + getGattsIf(), + addr, // address + 1 // direct connection + ); + if (errRc != ESP_OK) { + log_e("esp_ble_gattc_open: rc=%d %s", errRc, GeneralUtils::errorToString(errRc)); + return false; + } + + uint32_t rc = m_semaphoreOpenEvt.wait("connect"); // Wait for the connection to complete. + log_v("<< connect(), rc=%d", rc == ESP_GATT_OK); + return rc == ESP_GATT_OK; +} // connect + +/** + * Update connection parameters can be called only after connection has been established + */ +void BLEServer::updateConnParams(esp_bd_addr_t remote_bda, uint16_t minInterval, uint16_t maxInterval, uint16_t latency, uint16_t timeout) { + esp_ble_conn_update_params_t conn_params; + memcpy(conn_params.bda, remote_bda, sizeof(esp_bd_addr_t)); + conn_params.latency = latency; + conn_params.max_int = maxInterval; // max_int = 0x20*1.25ms = 40ms + conn_params.min_int = minInterval; // min_int = 0x10*1.25ms = 20ms + conn_params.timeout = timeout; // timeout = 400*10ms = 4000ms + esp_ble_gap_update_conn_params(&conn_params); +} + +void BLEServer::disconnect(uint16_t connId) { + esp_ble_gatts_close(m_gatts_if, connId); +} + uint16_t BLEServer::getGattsIf() { return m_gatts_if; } @@ -288,140 +522,450 @@ void BLEServer::registerApp(uint16_t m_appId) { log_v("<< registerApp"); } // registerApp +// Bluedroid callbacks + +void BLEServerCallbacks::onConnect(BLEServer *pServer, esp_ble_gatts_cb_param_t *param) { + log_d("BLEServerCallbacks", ">> onConnect(): Default"); + log_d("BLEServerCallbacks", "Device: %s", BLEDevice::toString().c_str()); + log_d("BLEServerCallbacks", "<< onConnect()"); +} // onConnect + +void BLEServerCallbacks::onDisconnect(BLEServer *pServer, esp_ble_gatts_cb_param_t *param) { + log_d("BLEServerCallbacks", ">> onDisconnect(): Default"); + log_d("BLEServerCallbacks", "Device: %s", BLEDevice::toString().c_str()); + log_d("BLEServerCallbacks", "<< onDisconnect()"); +} // onDisconnect + +void BLEServerCallbacks::onMtuChanged(BLEServer *pServer, esp_ble_gatts_cb_param_t *param) { + [[maybe_unused]] + uint16_t mtu = param->mtu.mtu; + log_d("BLEServerCallbacks", ">> onMtuChanged(): Default"); + log_d("BLEServerCallbacks", "Device: %s MTU: %d", BLEDevice::toString().c_str(), mtu); + log_d("BLEServerCallbacks", "<< onMtuChanged()"); +} // onMtuChanged + +#endif + +/*************************************************************************** + * NimBLE functions * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) + +uint16_t BLEServer::getHandle() { + return getConnId(); +} + /** - * @brief Set the server callbacks. + * @brief Resets the GATT server, used when services are added/removed after initialization. + */ +void BLEServer::resetGATT() { + if (getConnectedCount() > 0) { + return; + } + + BLEDevice::stopAdvertising(); + ble_gatts_reset(); + ble_svc_gap_init(); + ble_svc_gatt_init(); + + BLEService *svc = m_serviceMap.getFirst(); + while (svc != nullptr) { + if (svc->m_removed > 0) { + if (svc->m_removed == NIMBLE_ATT_REMOVE_DELETE) { + m_serviceMap.removeService(svc); + delete svc; + } + } else { + svc->start(); + } + + svc = m_serviceMap.getNext(); + } + + m_svcChanged = false; + m_gattsStarted = false; +} + +/** + * @brief Handle a GATT Server Event. * - * As a %BLE server operates, it will generate server level events such as a new client connecting or a previous client - * disconnecting. This function can be called to register a callback handler that will be invoked when these - * events are detected. + * @param [in] event + * @param [in] gatts_if + * @param [in] param * - * @param [in] pCallbacks The callbacks to be invoked. */ -void BLEServer::setCallbacks(BLEServerCallbacks *pCallbacks) { - m_pServerCallbacks = pCallbacks; -} // setCallbacks +int BLEServer::handleGATTServerEvent(struct ble_gap_event *event, void *arg) { + BLEServer *server = (BLEServer *)arg; + log_v(">> handleGAPEvent: %s", BLEUtils::gapEventToString(event->type)); + int rc = 0; + struct ble_gap_conn_desc desc; + + switch (event->type) { + case BLE_GAP_EVENT_CONNECT: + { + if (event->connect.status != 0) { + /* Connection failed; resume advertising */ + log_e("Connection failed"); + BLEDevice::startAdvertising(); + } else { + server->m_connId = event->connect.conn_handle; + server->addPeerDevice((void *)server, false, event->connect.conn_handle); + rc = ble_gap_conn_find(event->connect.conn_handle, &desc); + if (rc != 0) { + return 0; + } + + if (server->m_pServerCallbacks != nullptr) { + server->m_pServerCallbacks->onConnect(server); + server->m_pServerCallbacks->onConnect(server, &desc); + } + + server->m_connectedCount++; + } -/* - * Remove service - */ -void BLEServer::removeService(BLEService *service) { - service->stop(); - service->executeDelete(); - m_serviceMap.removeService(service); + return 0; + } // BLE_GAP_EVENT_CONNECT + + case BLE_GAP_EVENT_DISCONNECT: + { + // If Host reset tell the device now before returning to prevent + // any errors caused by calling host functions before resyncing. + switch (event->disconnect.reason) { + case BLE_HS_ETIMEOUT_HCI: + case BLE_HS_EOS: + case BLE_HS_ECONTROLLER: + case BLE_HS_ENOTSYNCED: + log_d("Disconnect - host reset, rc=%d", event->disconnect.reason); + BLEDevice::onReset(event->disconnect.reason); + break; + default: break; + } + + if (server->removePeerDevice(event->disconnect.conn.conn_handle, false)) { + server->m_connectedCount--; + } + + if (server->m_svcChanged) { + server->resetGATT(); + } + + if (server->m_pServerCallbacks != nullptr) { + server->m_pServerCallbacks->onDisconnect(server); + server->m_pServerCallbacks->onDisconnect(server, &event->disconnect.conn); + } + + return 0; + } // BLE_GAP_EVENT_DISCONNECT + + case BLE_GAP_EVENT_SUBSCRIBE: + { + log_i("subscribe event; attr_handle=%d, subscribed: %s", event->subscribe.attr_handle, (event->subscribe.cur_notify ? "true" : "false")); + + for (auto &it : server->m_notifyChrVec) { + if (it->getHandle() == event->subscribe.attr_handle) { + uint16_t properties = it->getProperties(); + if ((properties & BLE_GATT_CHR_F_READ_AUTHEN) || (properties & BLE_GATT_CHR_F_READ_AUTHOR) || (properties & BLE_GATT_CHR_F_READ_ENC)) { + rc = ble_gap_conn_find(event->subscribe.conn_handle, &desc); + if (rc != 0) { + break; + } + + if (!desc.sec_state.encrypted) { + BLESecurity::startSecurity(event->subscribe.conn_handle); + } + } + + it->setSubscribe(event); + break; + } + } + + return 0; + } // BLE_GAP_EVENT_SUBSCRIBE + + case BLE_GAP_EVENT_MTU: + { + log_i("mtu update event; conn_handle=%d mtu=%d", event->mtu.conn_handle, event->mtu.value); + rc = ble_gap_conn_find(event->mtu.conn_handle, &desc); + if (rc != 0) { + return 0; + } + + if (server->m_pServerCallbacks != nullptr) { + server->m_pServerCallbacks->onMtuChanged(server, &desc, event->mtu.value); + } + return 0; + } // BLE_GAP_EVENT_MTU + + case BLE_GAP_EVENT_NOTIFY_TX: + { + BLECharacteristic *pChar = nullptr; + + for (auto &it : server->m_notifyChrVec) { + if (it->getHandle() == event->notify_tx.attr_handle) { + pChar = it; + } + } + + if (pChar == nullptr) { + return 0; + } + + BLECharacteristicCallbacks::Status statusRC; + + if (event->notify_tx.indication) { + if (event->notify_tx.status != 0) { + if (event->notify_tx.status == BLE_HS_EDONE) { + statusRC = BLECharacteristicCallbacks::Status::SUCCESS_INDICATE; + } else if (rc == BLE_HS_ETIMEOUT) { + statusRC = BLECharacteristicCallbacks::Status::ERROR_INDICATE_TIMEOUT; + } else { + statusRC = BLECharacteristicCallbacks::Status::ERROR_INDICATE_FAILURE; + } + } else { + return 0; + } + + server->clearIndicateWait(event->notify_tx.conn_handle); + } else { + if (event->notify_tx.status == 0) { + statusRC = BLECharacteristicCallbacks::Status::SUCCESS_NOTIFY; + } else { + statusRC = BLECharacteristicCallbacks::Status::ERROR_GATT; + } + } + + pChar->m_pCallbacks->onStatus(pChar, statusRC, event->notify_tx.status); + + return 0; + } // BLE_GAP_EVENT_NOTIFY_TX + + case BLE_GAP_EVENT_ADV_COMPLETE: + { + log_d("Advertising Complete"); + BLEDevice::getAdvertising()->advCompleteCB(); + return 0; + } + + case BLE_GAP_EVENT_CONN_UPDATE: + { + log_d("Connection parameters updated."); + return 0; + } // BLE_GAP_EVENT_CONN_UPDATE + + case BLE_GAP_EVENT_REPEAT_PAIRING: + { + /* We already have a bond with the peer, but it is attempting to + * establish a new secure link. This app sacrifices security for + * convenience: just throw away the old bond and accept the new link. + */ + + /* Delete the old bond. */ + rc = ble_gap_conn_find(event->repeat_pairing.conn_handle, &desc); + if (rc != 0) { + return BLE_GAP_REPEAT_PAIRING_IGNORE; + } + + ble_store_util_delete_peer(&desc.peer_id_addr); + + /* Return BLE_GAP_REPEAT_PAIRING_RETRY to indicate that the host should + * continue with the pairing operation. + */ + return BLE_GAP_REPEAT_PAIRING_RETRY; + } // BLE_GAP_EVENT_REPEAT_PAIRING + + case BLE_GAP_EVENT_ENC_CHANGE: + { + rc = ble_gap_conn_find(event->enc_change.conn_handle, &desc); + if (rc != 0) { + return BLE_ATT_ERR_INVALID_HANDLE; + } + + if (BLEDevice::m_securityCallbacks != nullptr) { + BLEDevice::m_securityCallbacks->onAuthenticationComplete(&desc); + } else if (server->m_pServerCallbacks != nullptr) { + server->m_pServerCallbacks->onAuthenticationComplete(&desc); + } + + return 0; + } // BLE_GAP_EVENT_ENC_CHANGE + + case BLE_GAP_EVENT_PASSKEY_ACTION: + { + struct ble_sm_io pkey = {0, 0}; + + if (event->passkey.params.action == BLE_SM_IOACT_DISP) { + pkey.action = event->passkey.params.action; + // backward compatibility + pkey.passkey = BLESecurity::m_passkey; + // if the (static)passkey is the default, check the callback for custom value + // both values default to the same. + if (pkey.passkey == BLE_SM_DEFAULT_PASSKEY) { + if (server->m_pServerCallbacks != nullptr) { + pkey.passkey = server->m_pServerCallbacks->onPassKeyRequest(); + } + } + rc = ble_sm_inject_io(event->passkey.conn_handle, &pkey); + log_d("BLE_SM_IOACT_DISP; ble_sm_inject_io result: %d", rc); + + } else if (event->passkey.params.action == BLE_SM_IOACT_NUMCMP) { + log_d("Passkey on device's display: %d", event->passkey.params.numcmp); + pkey.action = event->passkey.params.action; + // Compatibility only - Do not use, should be removed the in future + if (BLEDevice::m_securityCallbacks != nullptr) { + pkey.numcmp_accept = BLEDevice::m_securityCallbacks->onConfirmPIN(event->passkey.params.numcmp); + } else if (server->m_pServerCallbacks != nullptr) { + pkey.numcmp_accept = server->m_pServerCallbacks->onConfirmPIN(event->passkey.params.numcmp); + } + + rc = ble_sm_inject_io(event->passkey.conn_handle, &pkey); + log_d("BLE_SM_IOACT_NUMCMP; ble_sm_inject_io result: %d", rc); + + //TODO: Handle out of band pairing + } else if (event->passkey.params.action == BLE_SM_IOACT_OOB) { + static uint8_t tem_oob[16] = {0}; + pkey.action = event->passkey.params.action; + for (int i = 0; i < 16; i++) { + pkey.oob[i] = tem_oob[i]; + } + rc = ble_sm_inject_io(event->passkey.conn_handle, &pkey); + log_d("BLE_SM_IOACT_OOB; ble_sm_inject_io result: %d", rc); + } else if (event->passkey.params.action == BLE_SM_IOACT_INPUT) { + log_d("Enter the passkey"); + pkey.action = event->passkey.params.action; + + // Compatibility only - Do not use, should be removed the in future + if (BLEDevice::m_securityCallbacks != nullptr) { + pkey.passkey = BLEDevice::m_securityCallbacks->onPassKeyRequest(); + } else if (server->m_pServerCallbacks != nullptr) { + pkey.passkey = server->m_pServerCallbacks->onPassKeyRequest(); + } + + rc = ble_sm_inject_io(event->passkey.conn_handle, &pkey); + log_d("BLE_SM_IOACT_INPUT; ble_sm_inject_io result: %d", rc); + + } else if (event->passkey.params.action == BLE_SM_IOACT_NONE) { + log_d("No passkey action required"); + } + + log_d("<< handleGATTServerEvent"); + return 0; + } // BLE_GAP_EVENT_PASSKEY_ACTION + + default: break; + } + + log_d("<< handleGATTServerEvent"); + return 0; } /** - * @brief Start advertising. - * - * Start the server advertising its existence. This is a convenience function and is equivalent to - * retrieving the advertising object and invoking start upon it. + * @brief Request an Update the connection parameters: + * * Can only be used after a connection has been established. + * @param [in] conn_handle The connection handle of the peer to send the request to. + * @param [in] minInterval The minimum connection interval in 1.25ms units. + * @param [in] maxInterval The maximum connection interval in 1.25ms units. + * @param [in] latency The number of packets allowed to skip (extends max interval). + * @param [in] timeout The timeout time in 10ms units before disconnecting. */ -void BLEServer::startAdvertising() { - log_v(">> startAdvertising"); - BLEDevice::startAdvertising(); - log_v("<< startAdvertising"); -} // startAdvertising +void BLEServer::updateConnParams(uint16_t conn_handle, uint16_t minInterval, uint16_t maxInterval, uint16_t latency, uint16_t timeout) { + ble_gap_upd_params params; + + params.latency = latency; + params.itvl_max = maxInterval; // max_int = 0x20*1.25ms = 40ms + params.itvl_min = minInterval; // min_int = 0x10*1.25ms = 20ms + params.supervision_timeout = timeout; // timeout = 400*10ms = 4000ms + params.min_ce_len = BLE_GAP_INITIAL_CONN_MIN_CE_LEN; // Minimum length of connection event in 0.625ms units + params.max_ce_len = BLE_GAP_INITIAL_CONN_MAX_CE_LEN; // Maximum length of connection event in 0.625ms units + + int rc = ble_gap_update_params(conn_handle, ¶ms); + if (rc != 0) { + log_e("Update params error: %d, %s", rc, BLEUtils::returnCodeToString(rc)); + } +} // updateConnParams + +bool BLEServer::setIndicateWait(uint16_t conn_handle) { + for (auto i = 0; i < CONFIG_BT_NIMBLE_MAX_CONNECTIONS; i++) { + if (m_indWait[i] == conn_handle) { + return false; + } + } + + return true; +} + +void BLEServer::clearIndicateWait(uint16_t conn_handle) { + for (auto i = 0; i < CONFIG_BT_NIMBLE_MAX_CONNECTIONS; i++) { + if (m_indWait[i] == conn_handle) { + m_indWait[i] = BLE_HS_CONN_HANDLE_NONE; + return; + } + } +} /** - * Allow to connect GATT server to peer device - * Probably can be used in ANCS for iPhone + * @brief Disconnect the specified client with optional reason. + * @param [in] connId Connection Id of the client to disconnect. + * @param [in] reason code for disconnecting. + * @return NimBLE host return code. */ -bool BLEServer::connect(BLEAddress address) { - esp_bd_addr_t addr; - memcpy(&addr, address.getNative(), 6); - // Perform the open connection request against the target BLE Server. - m_semaphoreOpenEvt.take("connect"); - esp_err_t errRc = ::esp_ble_gatts_open( - getGattsIf(), - addr, // address - 1 // direct connection - ); - if (errRc != ESP_OK) { - log_e("esp_ble_gattc_open: rc=%d %s", errRc, GeneralUtils::errorToString(errRc)); - return false; +int BLEServer::disconnect(uint16_t connId, uint8_t reason) { + log_d(">> disconnect()"); + + int rc = ble_gap_terminate(connId, reason); + if (rc != 0) { + log_e("ble_gap_terminate failed: rc=%d %s", rc, BLEUtils::returnCodeToString(rc)); } - uint32_t rc = m_semaphoreOpenEvt.wait("connect"); // Wait for the connection to complete. - log_v("<< connect(), rc=%d", rc == ESP_GATT_OK); - return rc == ESP_GATT_OK; -} // connect + log_d("<< disconnect()"); + return rc; +} // disconnect -void BLEServerCallbacks::onConnect(BLEServer *pServer) { - log_d("BLEServerCallbacks", ">> onConnect(): Default"); - log_d("BLEServerCallbacks", "Device: %s", BLEDevice::toString().c_str()); - log_d("BLEServerCallbacks", "<< onConnect()"); -} // onConnect +/** + * @brief Set the service changed flag + */ +void BLEServer::serviceChanged() { + if (m_gattsStarted) { + m_svcChanged = true; + } +} // serviceChanged -void BLEServerCallbacks::onConnect(BLEServer *pServer, esp_ble_gatts_cb_param_t *param) { +// NimBLE callbacks + +void BLEServerCallbacks::onConnect(BLEServer *pServer, struct ble_gap_conn_desc *desc) { log_d("BLEServerCallbacks", ">> onConnect(): Default"); log_d("BLEServerCallbacks", "Device: %s", BLEDevice::toString().c_str()); log_d("BLEServerCallbacks", "<< onConnect()"); } // onConnect -void BLEServerCallbacks::onDisconnect(BLEServer *pServer) { +void BLEServerCallbacks::onDisconnect(BLEServer *pServer, struct ble_gap_conn_desc *desc) { log_d("BLEServerCallbacks", ">> onDisconnect(): Default"); log_d("BLEServerCallbacks", "Device: %s", BLEDevice::toString().c_str()); log_d("BLEServerCallbacks", "<< onDisconnect()"); } // onDisconnect -void BLEServerCallbacks::onDisconnect(BLEServer *pServer, esp_ble_gatts_cb_param_t *param) { - log_d("BLEServerCallbacks", ">> onDisconnect(): Default"); - log_d("BLEServerCallbacks", "Device: %s", BLEDevice::toString().c_str()); - log_d("BLEServerCallbacks", "<< onDisconnect()"); -} // onDisconnect - -void BLEServerCallbacks::onMtuChanged(BLEServer *pServer, esp_ble_gatts_cb_param_t *param) { +void BLEServerCallbacks::onMtuChanged(BLEServer *pServer, ble_gap_conn_desc *desc, uint16_t mtu) { log_d("BLEServerCallbacks", ">> onMtuChanged(): Default"); - log_d("BLEServerCallbacks", "Device: %s MTU: %d", BLEDevice::toString().c_str(), param->mtu.mtu); + log_d("BLEServerCallbacks", "Device: %s MTU: %d", BLEDevice::toString().c_str(), mtu); log_d("BLEServerCallbacks", "<< onMtuChanged()"); } // onMtuChanged -/* multi connect support */ -/* TODO do some more tweaks */ -void BLEServer::updatePeerMTU(uint16_t conn_id, uint16_t mtu) { - // set mtu in conn_status_t - const std::map::iterator it = m_connectedServersMap.find(conn_id); - if (it != m_connectedServersMap.end()) { - it->second.mtu = mtu; - std::swap(m_connectedServersMap[conn_id], it->second); - } +uint32_t BLEServerCallbacks::onPassKeyRequest() { + log_d("BLEServerCallbacks", "onPassKeyRequest: default: 123456"); + return 123456; } -std::map BLEServer::getPeerDevices(bool _client) { - return m_connectedServersMap; +void BLEServerCallbacks::onAuthenticationComplete(ble_gap_conn_desc *) { + log_d("BLEServerCallbacks", "onAuthenticationComplete: default"); } -uint16_t BLEServer::getPeerMTU(uint16_t conn_id) { - return m_connectedServersMap.find(conn_id)->second.mtu; +bool BLEServerCallbacks::onConfirmPIN(uint32_t pin) { + log_d("BLEServerCallbacks", "onConfirmPIN: default: true"); + return true; } -void BLEServer::addPeerDevice(void *peer, bool _client, uint16_t conn_id) { - conn_status_t status = {.peer_device = peer, .connected = true, .mtu = 23}; - - m_connectedServersMap.insert(std::pair(conn_id, status)); -} - -bool BLEServer::removePeerDevice(uint16_t conn_id, bool _client) { - return m_connectedServersMap.erase(conn_id) > 0; -} -/* multi connect support */ - -/** - * Update connection parameters can be called only after connection has been established - */ -void BLEServer::updateConnParams(esp_bd_addr_t remote_bda, uint16_t minInterval, uint16_t maxInterval, uint16_t latency, uint16_t timeout) { - esp_ble_conn_update_params_t conn_params; - memcpy(conn_params.bda, remote_bda, sizeof(esp_bd_addr_t)); - conn_params.latency = latency; - conn_params.max_int = maxInterval; // max_int = 0x20*1.25ms = 40ms - conn_params.min_int = minInterval; // min_int = 0x10*1.25ms = 20ms - conn_params.timeout = timeout; // timeout = 400*10ms = 4000ms - esp_ble_gap_update_conn_params(&conn_params); -} - -void BLEServer::disconnect(uint16_t connId) { - esp_ble_gatts_close(m_gatts_if, connId); -} +#endif // CONFIG_NIMBLE_ENABLED -#endif /* CONFIG_BLUEDROID_ENABLED */ +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ diff --git a/libraries/BLE/src/BLEServer.h b/libraries/BLE/src/BLEServer.h index aa10f2210fa..865d1046312 100644 --- a/libraries/BLE/src/BLEServer.h +++ b/libraries/BLE/src/BLEServer.h @@ -3,6 +3,10 @@ * * Created on: Apr 16, 2017 * Author: kolban + * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on kolban's and h2zero's work) + * Description: Added support for NimBLE */ #ifndef COMPONENTS_CPP_UTILS_BLESERVER_H_ @@ -11,13 +15,16 @@ #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) -#include +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) + +/***************************************************************************** + * Common includes * + *****************************************************************************/ #include #include -// #include "BLEDevice.h" - +#include "BLEDevice.h" +#include "BLEConnInfo.h" #include "BLEUUID.h" #include "BLEAdvertising.h" #include "BLECharacteristic.h" @@ -25,24 +32,51 @@ #include "BLESecurity.h" #include "RTOS.h" #include "BLEAddress.h" +#include "BLEUtils.h" +#include "BLEUtils.h" + +/***************************************************************************** + * Bluedroid includes * + *****************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) +#include +#endif + +/***************************************************************************** + * NimBLE includes and definitions * + *****************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) +#include +#define ESP_GATT_IF_NONE BLE_HS_CONN_HANDLE_NONE +#define NIMBLE_ATT_REMOVE_HIDE 1 +#define NIMBLE_ATT_REMOVE_DELETE 2 +#endif + +/***************************************************************************** + * Forward declarations * + *****************************************************************************/ class BLEServerCallbacks; -/* TODO possibly refactor this struct */ -typedef struct { - void *peer_device; // peer device BLEClient or BLEServer - maybe its better to have 2 structures or union here - bool connected; // do we need it? - uint16_t mtu; // every peer device negotiate own mtu -} conn_status_t; +class BLEService; +class BLECharacteristic; +class BLEDevice; +class BLESecurity; +class BLEAdvertising; /** - * @brief A data structure that manages the %BLE servers owned by a BLE server. + * @brief A data structure that manages the %BLE services owned by a BLE server. */ class BLEServiceMap { public: + /*************************************************************************** + * Common public declarations * + ***************************************************************************/ + BLEService *getByHandle(uint16_t handle); BLEService *getByUUID(const char *uuid); BLEService *getByUUID(BLEUUID uuid, uint8_t inst_id = 0); - void handleGATTServerEvent(esp_gatts_cb_event_t event, esp_gatt_if_t gatts_if, esp_ble_gatts_cb_param_t *param); void setByHandle(uint16_t handle, BLEService *service); void setByUUID(const char *uuid, BLEService *service); void setByUUID(BLEUUID uuid, BLEService *service); @@ -52,7 +86,19 @@ class BLEServiceMap { void removeService(BLEService *service); int getRegisteredServiceCount(); + /*************************************************************************** + * Bluedroid public declarations * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) + void handleGATTServerEvent(esp_gatts_cb_event_t event, esp_gatt_if_t gatts_if, esp_ble_gatts_cb_param_t *param); +#endif + private: + /*************************************************************************** + * Common private properties * + ***************************************************************************/ + std::map m_handleMap; std::map m_uuidMap; std::map::iterator m_iterator; @@ -63,6 +109,16 @@ class BLEServiceMap { */ class BLEServer { public: + /*************************************************************************** + * Common public properties * + ***************************************************************************/ + + uint16_t m_appId; + + /*************************************************************************** + * Common public declarations * + ***************************************************************************/ + uint32_t getConnectedCount(); BLEService *createService(const char *uuid); BLEService *createService(BLEUUID uuid, uint32_t numHandles = 15, uint8_t inst_id = 0); @@ -72,12 +128,9 @@ class BLEServer { void removeService(BLEService *service); BLEService *getServiceByUUID(const char *uuid); BLEService *getServiceByUUID(BLEUUID uuid); - bool connect(BLEAddress address); - void disconnect(uint16_t connId); - uint16_t m_appId; - void updateConnParams(esp_bd_addr_t remote_bda, uint16_t minInterval, uint16_t maxInterval, uint16_t latency, uint16_t timeout); + void start(); - /* multi connection support */ + // Connection management functions std::map getPeerDevices(bool client); void addPeerDevice(void *peer, bool is_client, uint16_t conn_id); bool removePeerDevice(uint16_t conn_id, bool client); @@ -86,28 +139,95 @@ class BLEServer { uint16_t getPeerMTU(uint16_t conn_id); uint16_t getConnId(); + /*************************************************************************** + * Bluedroid public declarations * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) + bool connect(BLEAddress address); + void updateConnParams(esp_bd_addr_t remote_bda, uint16_t minInterval, uint16_t maxInterval, uint16_t latency, uint16_t timeout); + void disconnect(uint16_t connId); +#endif + + /*************************************************************************** + * NimBLE public declarations * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) + uint16_t getHandle(); + void updateConnParams(uint16_t conn_handle, uint16_t minInterval, uint16_t maxInterval, uint16_t latency, uint16_t timeout); + int disconnect(uint16_t connId, uint8_t reason = BLE_ERR_REM_USER_CONN_TERM); +#endif + private: - BLEServer(); friend class BLEService; friend class BLECharacteristic; friend class BLEDevice; - esp_ble_adv_data_t m_adv_data; - // BLEAdvertising m_bleAdvertising; + friend class BLESecurity; + friend class BLEAdvertising; + + /*************************************************************************** + * Common private properties * + ***************************************************************************/ + uint16_t m_connId; uint32_t m_connectedCount; - uint16_t m_gatts_if; + bool m_gattsStarted; std::map m_connectedServersMap; - FreeRTOS::Semaphore m_semaphoreRegisterAppEvt = FreeRTOS::Semaphore("RegisterAppEvt"); FreeRTOS::Semaphore m_semaphoreCreateEvt = FreeRTOS::Semaphore("CreateEvt"); FreeRTOS::Semaphore m_semaphoreOpenEvt = FreeRTOS::Semaphore("OpenEvt"); BLEServiceMap m_serviceMap; BLEServerCallbacks *m_pServerCallbacks = nullptr; + /*************************************************************************** + * Bluedroid private properties * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) + uint16_t m_gatts_if; + esp_ble_adv_data_t m_adv_data; +#endif + + /*************************************************************************** + * NimBLE private properties * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) + bool m_svcChanged; + uint16_t m_indWait[CONFIG_BT_NIMBLE_MAX_CONNECTIONS]; + std::vector m_notifyChrVec; + ble_hs_adv_fields m_adv_data; +#endif + + /*************************************************************************** + * Common private declarations * + ***************************************************************************/ + + BLEServer(); void createApp(uint16_t appId); + + /*************************************************************************** + * Bluedroid private declarations * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) uint16_t getGattsIf(); - void handleGATTServerEvent(esp_gatts_cb_event_t event, esp_gatt_if_t gatts_if, esp_ble_gatts_cb_param_t *param); void registerApp(uint16_t); + void handleGATTServerEvent(esp_gatts_cb_event_t event, esp_gatt_if_t gatts_if, esp_ble_gatts_cb_param_t *param); +#endif + + /*************************************************************************** + * NimBLE private declarations * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) + void serviceChanged(); + void resetGATT(); + bool setIndicateWait(uint16_t conn_handle); + void clearIndicateWait(uint16_t conn_handle); + static int handleGATTServerEvent(struct ble_gap_event *event, void *arg); +#endif }; // BLEServer /** @@ -115,37 +235,38 @@ class BLEServer { */ class BLEServerCallbacks { public: + /*************************************************************************** + * Common public declarations * + ***************************************************************************/ + virtual ~BLEServerCallbacks(){}; - /** - * @brief Handle a new client connection. - * - * When a new client connects, we are invoked. - * - * @param [in] pServer A reference to the %BLE server that received the client connection. - */ virtual void onConnect(BLEServer *pServer); - virtual void onConnect(BLEServer *pServer, esp_ble_gatts_cb_param_t *param); - /** - * @brief Handle an existing client disconnection. - * - * When an existing client disconnects, we are invoked. - * - * @param [in] pServer A reference to the %BLE server that received the existing client disconnection. - */ virtual void onDisconnect(BLEServer *pServer); - virtual void onDisconnect(BLEServer *pServer, esp_ble_gatts_cb_param_t *param); - /** - * @brief Handle a new client connection. - * - * When the MTU changes this method is invoked. - * - * @param [in] pServer A reference to the %BLE server that received the client connection. - * @param [in] param A reference to esp_ble_gatts_cb_param_t. - */ + /*************************************************************************** + * Bluedroid public declarations * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) + virtual void onConnect(BLEServer *pServer, esp_ble_gatts_cb_param_t *param); + virtual void onDisconnect(BLEServer *pServer, esp_ble_gatts_cb_param_t *param); virtual void onMtuChanged(BLEServer *pServer, esp_ble_gatts_cb_param_t *param); +#endif + + /*************************************************************************** + * NimBLE public declarations * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) + virtual void onConnect(BLEServer *pServer, ble_gap_conn_desc *desc); + virtual void onDisconnect(BLEServer *pServer, ble_gap_conn_desc *desc); + virtual void onMtuChanged(BLEServer *pServer, ble_gap_conn_desc *desc, uint16_t mtu); + virtual uint32_t onPassKeyRequest(); + virtual void onAuthenticationComplete(ble_gap_conn_desc *desc); + virtual bool onConfirmPIN(uint32_t pin); +#endif }; // BLEServerCallbacks -#endif /* CONFIG_BLUEDROID_ENABLED */ +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ #endif /* COMPONENTS_CPP_UTILS_BLESERVER_H_ */ diff --git a/libraries/BLE/src/BLEService.cpp b/libraries/BLE/src/BLEService.cpp index 58c5d4eb9c3..60449719b6a 100644 --- a/libraries/BLE/src/BLEService.cpp +++ b/libraries/BLE/src/BLEService.cpp @@ -3,6 +3,10 @@ * * Created on: Mar 25, 2017 * Author: kolban + * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on kolban's and h2zero's work) + * Description: Added support for NimBLE */ // A service is identified by a UUID. A service is also the container for one or more characteristics. @@ -11,10 +15,13 @@ #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) -#include -#include +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) +/***************************************************************************** + * Common includes * + *****************************************************************************/ + +#include #include #include #include @@ -25,8 +32,24 @@ #include "GeneralUtils.h" #include "esp32-hal-log.h" +/***************************************************************************** + * Bluedroid includes * + *****************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) +#include +#endif + +/***************************************************************************** + * Common definitions * + *****************************************************************************/ + #define NULL_HANDLE (0xffff) +/***************************************************************************** + * Common functions * + *****************************************************************************/ + /** * @brief Construct an instance of the BLEService * @param [in] uuid The UUID of the service. @@ -44,10 +67,32 @@ BLEService::BLEService(BLEUUID uuid, uint16_t numHandles) { m_handle = NULL_HANDLE; m_pServer = nullptr; //m_serializeMutex.setName("BLEService"); - m_lastCreatedCharacteristic = nullptr; m_numHandles = numHandles; +#ifdef CONFIG_BLUEDROID_ENABLED + m_lastCreatedCharacteristic = nullptr; +#endif +#if defined(CONFIG_NIMBLE_ENABLED) + m_pSvcDef = nullptr; +#endif } // BLEService +BLEService::~BLEService() { +#if defined(CONFIG_NIMBLE_ENABLED) + if (m_pSvcDef != nullptr) { + if (m_pSvcDef->characteristics != nullptr) { + for (int i = 0; m_pSvcDef->characteristics[i].uuid != NULL; ++i) { + if (m_pSvcDef->characteristics[i].descriptors) { + delete (m_pSvcDef->characteristics[i].descriptors); + } + } + delete (m_pSvcDef->characteristics); + } + + delete (m_pSvcDef); + } +#endif +} + /** * @brief Create the service. * Create the service. @@ -56,8 +101,9 @@ BLEService::BLEService(BLEUUID uuid, uint16_t numHandles) { */ void BLEService::executeCreate(BLEServer *pServer) { - log_v(">> executeCreate() - Creating service (esp_ble_gatts_create_service) service uuid: %s", getUUID().toString().c_str()); + log_v(">> executeCreate() - Creating service with uuid: %s", getUUID().toString().c_str()); m_pServer = pServer; +#if defined(CONFIG_BLUEDROID_ENABLED) m_semaphoreCreateEvt.take("executeCreate"); // Take the mutex and release at event ESP_GATTS_CREATE_EVT esp_gatt_srvc_id_t srvc_id; @@ -73,6 +119,7 @@ void BLEService::executeCreate(BLEServer *pServer) { } m_semaphoreCreateEvt.wait("executeCreate"); +#endif log_v("<< executeCreate"); } // executeCreate @@ -83,6 +130,7 @@ void BLEService::executeCreate(BLEServer *pServer) { */ void BLEService::executeDelete() { +#if defined(CONFIG_BLUEDROID_ENABLED) log_v(">> executeDelete()"); m_semaphoreDeleteEvt.take("executeDelete"); // Take the mutex and release at event ESP_GATTS_DELETE_EVT @@ -95,6 +143,11 @@ void BLEService::executeDelete() { m_semaphoreDeleteEvt.wait("executeDelete"); log_v("<< executeDelete"); +#endif + +#if defined(CONFIG_NIMBLE_ENABLED) && CONFIG_BT_NIMBLE_DYNAMIC_SERVICE + ble_gatts_delete_svc(&m_uuid.getNative()->u); +#endif } // executeDelete /** @@ -114,49 +167,11 @@ BLEUUID BLEService::getUUID() { return m_uuid; } // getUUID -/** - * @brief Start the service. - * Here we wish to start the service which means that we will respond to partner requests about it. - * Starting a service also means that we can create the corresponding characteristics. - * @return Start the service. - */ -void BLEService::start() { - // We ask the BLE runtime to start the service and then create each of the characteristics. - // We start the service through its local handle which was returned in the ESP_GATTS_CREATE_EVT event - // obtained as a result of calling esp_ble_gatts_create_service(). - // - log_v(">> start(): Starting service (esp_ble_gatts_start_service): %s", toString().c_str()); - if (m_handle == NULL_HANDLE) { - log_e("<< !!! We attempted to start a service but don't know its handle!"); - return; - } - - BLECharacteristic *pCharacteristic = m_characteristicMap.getFirst(); - - while (pCharacteristic != nullptr) { - m_lastCreatedCharacteristic = pCharacteristic; - pCharacteristic->executeCreate(this); - - pCharacteristic = m_characteristicMap.getNext(); - } - // Start each of the characteristics ... these are found in the m_characteristicMap. - - m_semaphoreStartEvt.take("start"); - esp_err_t errRc = ::esp_ble_gatts_start_service(m_handle); - - if (errRc != ESP_OK) { - log_e("<< esp_ble_gatts_start_service: rc=%d %s", errRc, GeneralUtils::errorToString(errRc)); - return; - } - m_semaphoreStartEvt.wait("start"); - - log_v("<< start()"); -} // start - /** * @brief Stop the service. */ void BLEService::stop() { +#if defined(CONFIG_BLUEDROID_ENABLED) // We ask the BLE runtime to start the service and then create each of the characteristics. // We start the service through its local handle which was returned in the ESP_GATTS_CREATE_EVT event // obtained as a result of calling esp_ble_gatts_create_service(). @@ -176,13 +191,19 @@ void BLEService::stop() { m_semaphoreStopEvt.wait("stop"); log_v("<< stop()"); -} // start +#endif + +#if defined(CONFIG_NIMBLE_ENABLED) + log_w("NimBLE does not support stopping a service. Ignoring request."); +#endif +} // stop /** * @brief Set the handle associated with this service. * @param [in] handle The handle associated with the service. */ void BLEService::setHandle(uint16_t handle) { +#if defined(CONFIG_BLUEDROID_ENABLED) log_v(">> setHandle - Handle=0x%.2x, service UUID=%s)", handle, getUUID().toString().c_str()); if (m_handle != NULL_HANDLE) { log_e("!!! Handle is already set %.2x", m_handle); @@ -190,6 +211,11 @@ void BLEService::setHandle(uint16_t handle) { } m_handle = handle; log_v("<< setHandle"); +#endif + +#if defined(CONFIG_NIMBLE_ENABLED) + log_w("NimBLE does not support manually setting the handle of a service. Ignoring request."); +#endif } // setHandle /** @@ -213,14 +239,25 @@ void BLEService::addCharacteristic(BLECharacteristic *pCharacteristic) { log_d("Adding characteristic: uuid=%s to service: %s", pCharacteristic->getUUID().toString().c_str(), toString().c_str()); // Check that we don't add the same characteristic twice. - if (m_characteristicMap.getByUUID(pCharacteristic->getUUID()) != nullptr) { + BLECharacteristic *pExisting = m_characteristicMap.getByUUID(pCharacteristic->getUUID()); + if (pExisting != nullptr) { log_w("<< Adding a new characteristic with the same UUID as a previous one"); - //return; } - // Remember this characteristic in our map of characteristics. At this point, we can lookup by UUID - // but not by handle. The handle is allocated to us on the ESP_GATTS_ADD_CHAR_EVT. - m_characteristicMap.setByUUID(pCharacteristic, pCharacteristic->getUUID()); +#if defined(CONFIG_NIMBLE_ENABLED) + if (pExisting != nullptr) { + pExisting->m_removed = 0; + } else +#endif + { + // Remember this characteristic in our map of characteristics. At this point, we can lookup by UUID + // but not by handle. The handle is allocated to us on the ESP_GATTS_ADD_CHAR_EVT. + m_characteristicMap.setByUUID(pCharacteristic, pCharacteristic->getUUID()); + } + +#if defined(CONFIG_NIMBLE_ENABLED) + getServer()->serviceChanged(); +#endif log_v("<< addCharacteristic()"); } // addCharacteristic @@ -247,6 +284,54 @@ BLECharacteristic *BLEService::createCharacteristic(BLEUUID uuid, uint32_t prope return pCharacteristic; } // createCharacteristic +BLECharacteristic *BLEService::getCharacteristic(const char *uuid) { + return getCharacteristic(BLEUUID(uuid)); +} + +BLECharacteristic *BLEService::getCharacteristic(BLEUUID uuid) { + return m_characteristicMap.getByUUID(uuid); +} + +/** + * @brief Return a string representation of this service. + * A service is defined by: + * * Its UUID + * * Its handle + * @return A string representation of this service. + */ +String BLEService::toString() { + String res = "UUID: " + getUUID().toString(); + char hex[5]; + snprintf(hex, sizeof(hex), "%04x", getHandle()); + res += ", handle: 0x"; + res += hex; + return res; +} // toString + +/** + * @brief Get the BLE server associated with this service. + * @return The BLEServer associated with this service. + */ +BLEServer *BLEService::getServer() { + return m_pServer; +} // getServer + +/***************************************************************************** + * Bluedroid functions * + *****************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) +/** + * @brief Get the last created characteristic. + * It is lamentable that this function has to exist. It returns the last created characteristic. + * We need this because the descriptor API is built around the notion that a new descriptor, when created, + * is associated with the last characteristics created and we need that information. + * @return The last created characteristic. + */ +BLECharacteristic *BLEService::getLastCreatedCharacteristic() { + return m_lastCreatedCharacteristic; +} // getLastCreatedCharacteristic + /** * @brief Handle a GATTS server event. */ @@ -347,48 +432,222 @@ void BLEService::handleGATTServerEvent(esp_gatts_cb_event_t event, esp_gatt_if_t m_characteristicMap.handleGATTServerEvent(event, gatts_if, param); } // handleGATTServerEvent -BLECharacteristic *BLEService::getCharacteristic(const char *uuid) { - return getCharacteristic(BLEUUID(uuid)); -} - -BLECharacteristic *BLEService::getCharacteristic(BLEUUID uuid) { - return m_characteristicMap.getByUUID(uuid); -} - /** - * @brief Return a string representation of this service. - * A service is defined by: - * * Its UUID - * * Its handle - * @return A string representation of this service. + * @brief Start the service. + * Here we wish to start the service which means that we will respond to partner requests about it. + * Starting a service also means that we can create the corresponding characteristics. + * @return Start the service. */ -String BLEService::toString() { - String res = "UUID: " + getUUID().toString(); - char hex[5]; - snprintf(hex, sizeof(hex), "%04x", getHandle()); - res += ", handle: 0x"; - res += hex; - return res; -} // toString +bool BLEService::start() { + // We ask the BLE runtime to start the service and then create each of the characteristics. + // We start the service through its local handle which was returned in the ESP_GATTS_CREATE_EVT event + // obtained as a result of calling esp_ble_gatts_create_service(). + // + log_v(">> start(): Starting service (esp_ble_gatts_start_service): %s", toString().c_str()); + if (m_handle == NULL_HANDLE) { + log_e("<< !!! We attempted to start a service but don't know its handle!"); + return false; + } + + BLECharacteristic *pCharacteristic = m_characteristicMap.getFirst(); + + while (pCharacteristic != nullptr) { + m_lastCreatedCharacteristic = pCharacteristic; + pCharacteristic->executeCreate(this); + + pCharacteristic = m_characteristicMap.getNext(); + } + // Start each of the characteristics ... these are found in the m_characteristicMap. + + m_semaphoreStartEvt.take("start"); + esp_err_t errRc = ::esp_ble_gatts_start_service(m_handle); + + if (errRc != ESP_OK) { + log_e("<< esp_ble_gatts_start_service: rc=%d %s", errRc, GeneralUtils::errorToString(errRc)); + return false; + } + m_semaphoreStartEvt.wait("start"); + + log_v("<< start()"); + return true; +} // start + +#endif // CONFIG_BLUEDROID_ENABLED + +/***************************************************************************** + * NimBLE functions * + *****************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) /** - * @brief Get the last created characteristic. - * It is lamentable that this function has to exist. It returns the last created characteristic. - * We need this because the descriptor API is built around the notion that a new descriptor, when created, - * is associated with the last characteristics created and we need that information. - * @return The last created characteristic. + * @brief Remove a characteristic from the service. Check if the characteristic was already removed and if so, check if this + * is being called to delete the object and do so if requested. Otherwise, ignore the call and return. + * @param [in] pCharacteristic - The characteristic to remove. + * @param [in] deleteChr - If true, delete the characteristic. */ -BLECharacteristic *BLEService::getLastCreatedCharacteristic() { - return m_lastCreatedCharacteristic; -} // getLastCreatedCharacteristic +void BLEService::removeCharacteristic(BLECharacteristic *pCharacteristic, bool deleteChr) { + if (pCharacteristic->m_removed > 0) { + if (deleteChr) { + BLECharacteristic *pExisting = m_characteristicMap.getByUUID(pCharacteristic->getUUID()); + if (pExisting != nullptr) { + m_characteristicMap.removeCharacteristic(pExisting); + delete pExisting; + } + } + + return; + } + + pCharacteristic->m_removed = deleteChr ? NIMBLE_ATT_REMOVE_DELETE : NIMBLE_ATT_REMOVE_HIDE; + getServer()->serviceChanged(); +} /** - * @brief Get the BLE server associated with this service. - * @return The BLEServer associated with this service. + * @brief Builds the database of characteristics/descriptors for the service + * and registers it with the NimBLE stack. + * @return bool success/failure . */ -BLEServer *BLEService::getServer() { - return m_pServer; -} // getServer +bool BLEService::start() { + log_d(">> start(): Starting service: %s", toString().c_str()); + + // Rebuild the service definition if the server attributes have changed. + if (getServer()->m_svcChanged && m_pSvcDef != nullptr) { + if (m_pSvcDef[0].characteristics) { + if (m_pSvcDef[0].characteristics[0].descriptors) { + delete (m_pSvcDef[0].characteristics[0].descriptors); + } + delete (m_pSvcDef[0].characteristics); + } + delete (m_pSvcDef); + m_pSvcDef = nullptr; + } + + if (m_pSvcDef == nullptr) { + // Nimble requires an array of services to be sent to the api + // Since we are adding 1 at a time we create an array of 2 and set the type + // of the second service to 0 to indicate the end of the array. + ble_gatt_svc_def *svc = new ble_gatt_svc_def[2]{}; + ble_gatt_chr_def *pChr_a = nullptr; + ble_gatt_dsc_def *pDsc_a = nullptr; + + svc[0].type = BLE_GATT_SVC_TYPE_PRIMARY; + svc[0].uuid = (const ble_uuid_t *)&(m_uuid.getNative()->u); + svc[0].includes = nullptr; + + int removedCount = 0; + BLECharacteristic *pCharacteristic; + + pCharacteristic = m_characteristicMap.getFirst(); + while (pCharacteristic != nullptr) { + if (pCharacteristic->m_removed > 0) { + if (pCharacteristic->m_removed == NIMBLE_ATT_REMOVE_DELETE) { + m_characteristicMap.removeCharacteristic(pCharacteristic); + delete pCharacteristic; + } else { + ++removedCount; + } + } else { + pCharacteristic->executeCreate(this); + } + + pCharacteristic = m_characteristicMap.getNext(); + } + + size_t numChrs = m_characteristicMap.getRegisteredCharacteristicCount() - removedCount; + log_d("Adding %d characteristics for service %s", numChrs, toString().c_str()); + + if (!numChrs) { + svc[0].characteristics = nullptr; + } else { + // Nimble requires the last characteristic to have it's uuid = 0 to indicate the end + // of the characteristics for the service. We create 1 extra and set it to null + // for this purpose. + pChr_a = new ble_gatt_chr_def[numChrs + 1]{}; + uint8_t i = 0; + pCharacteristic = m_characteristicMap.getFirst(); + while (pCharacteristic != nullptr) { + if (pCharacteristic->m_removed <= 0) { + removedCount = 0; + BLEDescriptor *pDescriptor; + + pDescriptor = pCharacteristic->m_descriptorMap.getFirst(); + while (pDescriptor != nullptr) { + if (pDescriptor->m_removed > 0) { + if (pDescriptor->m_removed == NIMBLE_ATT_REMOVE_DELETE) { + pCharacteristic->m_descriptorMap.removeDescriptor(pDescriptor); + delete pDescriptor; + } else { + ++removedCount; + } + } + pDescriptor = pCharacteristic->m_descriptorMap.getNext(); + } + + size_t numDscs = pCharacteristic->m_descriptorMap.getRegisteredDescriptorCount() - removedCount; + log_d("Adding %d descriptors for characteristic %s", numDscs, pCharacteristic->getUUID().toString().c_str()); + + if (!numDscs) { + pChr_a[i].descriptors = nullptr; + } else { + // Must have last descriptor uuid = 0 so we have to create 1 extra + pDsc_a = new ble_gatt_dsc_def[numDscs + 1]{}; + uint8_t d = 0; + pDescriptor = pCharacteristic->m_descriptorMap.getFirst(); + while (pDescriptor != nullptr) { + if (pDescriptor->m_removed <= 0) { + pDsc_a[d].uuid = (const ble_uuid_t *)&(pDescriptor->m_bleUUID.getNative()->u); + pDsc_a[d].att_flags = pDescriptor->m_permissions; + pDsc_a[d].min_key_size = 0; + pDsc_a[d].access_cb = BLEDescriptor::handleGATTServerEvent; + pDsc_a[d].arg = pDescriptor; + ++d; + } + pDescriptor = pCharacteristic->m_descriptorMap.getNext(); + } + + pDsc_a[numDscs].uuid = nullptr; + pChr_a[i].descriptors = pDsc_a; + } + + pChr_a[i].uuid = (const ble_uuid_t *)&(pCharacteristic->m_bleUUID.getNative()->u); + pChr_a[i].access_cb = BLECharacteristic::handleGATTServerEvent; + pChr_a[i].arg = pCharacteristic; + pChr_a[i].flags = pCharacteristic->m_properties; + pChr_a[i].min_key_size = 0; + pChr_a[i].val_handle = &pCharacteristic->m_handle; + ++i; + } + + pCharacteristic = m_characteristicMap.getNext(); + } + + pChr_a[numChrs].uuid = nullptr; + svc[0].characteristics = pChr_a; + } + + // end of services must indicate to api with type = 0 + svc[1].type = 0; + m_pSvcDef = svc; + } + + int rc = ble_gatts_count_cfg((const ble_gatt_svc_def *)m_pSvcDef); + if (rc != 0) { + log_e("ble_gatts_count_cfg failed, rc= %d, %s", rc, BLEUtils::returnCodeToString(rc)); + return false; + } + + rc = ble_gatts_add_svcs((const ble_gatt_svc_def *)m_pSvcDef); + if (rc != 0) { + log_e("ble_gatts_add_svcs, rc= %d, %s", rc, BLEUtils::returnCodeToString(rc)); + return false; + } + + log_d("<< start()"); + return true; +} // start + +#endif -#endif /* CONFIG_BLUEDROID_ENABLED */ +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ diff --git a/libraries/BLE/src/BLEService.h b/libraries/BLE/src/BLEService.h index 61f867b2a02..67088b60310 100644 --- a/libraries/BLE/src/BLEService.h +++ b/libraries/BLE/src/BLEService.h @@ -3,6 +3,10 @@ * * Created on: Mar 25, 2017 * Author: kolban + * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on kolban's and h2zero's work) + * Description: Added support for NimBLE */ #ifndef COMPONENTS_CPP_UTILS_BLESERVICE_H_ @@ -11,15 +15,38 @@ #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) -#include +/***************************************************************************** + * Common includes * + *****************************************************************************/ #include "BLECharacteristic.h" #include "BLEServer.h" #include "BLEUUID.h" +#include "BLEUtils.h" #include "RTOS.h" +/***************************************************************************** + * Bluedroid includes * + *****************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) +#include +#endif + +/***************************************************************************** + * NimBLE includes * + *****************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) +#include +#endif + +/***************************************************************************** + * Forward declarations * + *****************************************************************************/ + class BLEServer; /** @@ -27,6 +54,10 @@ class BLEServer; */ class BLECharacteristicMap { public: + /*************************************************************************** + * Common public declarations * + ***************************************************************************/ + void setByUUID(BLECharacteristic *pCharacteristic, const char *uuid); void setByUUID(BLECharacteristic *pCharacteristic, BLEUUID uuid); void setByHandle(uint16_t handle, BLECharacteristic *pCharacteristic); @@ -36,9 +67,30 @@ class BLECharacteristicMap { BLECharacteristic *getFirst(); BLECharacteristic *getNext(); String toString(); + int getRegisteredCharacteristicCount(); + void removeCharacteristic(BLECharacteristic *characteristic); + + /*************************************************************************** + * Bluedroid public declarations * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) void handleGATTServerEvent(esp_gatts_cb_event_t event, esp_gatt_if_t gatts_if, esp_ble_gatts_cb_param_t *param); +#endif + + /*************************************************************************** + * NimBLE public declarations * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) + void handleGATTServerEvent(uint16_t conn_handle, uint16_t attr_handle, ble_gatt_access_ctxt *ctxt, void *arg); +#endif private: + /*************************************************************************** + * Common private properties * + ***************************************************************************/ + std::map m_uuidMap; std::map m_handleMap; std::map::iterator m_iterator; @@ -46,10 +98,19 @@ class BLECharacteristicMap { /** * @brief The model of a %BLE service. - * */ class BLEService { public: + /*************************************************************************** + * Common properties * + ***************************************************************************/ + + uint8_t m_instId = 0; + + /*************************************************************************** + * Common public declarations * + ***************************************************************************/ + void addCharacteristic(BLECharacteristic *pCharacteristic); BLECharacteristic *createCharacteristic(const char *uuid, uint32_t properties); BLECharacteristic *createCharacteristic(BLEUUID uuid, uint32_t properties); @@ -60,40 +121,81 @@ class BLEService { BLECharacteristic *getCharacteristic(BLEUUID uuid); BLEUUID getUUID(); BLEServer *getServer(); - void start(); + bool start(); void stop(); String toString(); uint16_t getHandle(); - uint8_t m_instId = 0; + + /*************************************************************************** + * NimBLE public declarations * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) + void removeCharacteristic(BLECharacteristic *pCharacteristic, bool deleteChr = false); +#endif private: - BLEService(const char *uuid, uint16_t numHandles); - BLEService(BLEUUID uuid, uint16_t numHandles); friend class BLEServer; friend class BLEServiceMap; friend class BLEDescriptor; friend class BLECharacteristic; friend class BLEDevice; + /*************************************************************************** + * Common private properties * + ***************************************************************************/ + BLECharacteristicMap m_characteristicMap; uint16_t m_handle; BLECharacteristic *m_lastCreatedCharacteristic = nullptr; BLEServer *m_pServer = nullptr; BLEUUID m_uuid; + FreeRTOS::Semaphore m_semaphoreStartEvt = FreeRTOS::Semaphore("StartEvt"); + uint16_t m_numHandles; + + /*************************************************************************** + * Bluedroid private properties * + ***************************************************************************/ +#if defined(CONFIG_BLUEDROID_ENABLED) FreeRTOS::Semaphore m_semaphoreCreateEvt = FreeRTOS::Semaphore("CreateEvt"); FreeRTOS::Semaphore m_semaphoreDeleteEvt = FreeRTOS::Semaphore("DeleteEvt"); - FreeRTOS::Semaphore m_semaphoreStartEvt = FreeRTOS::Semaphore("StartEvt"); FreeRTOS::Semaphore m_semaphoreStopEvt = FreeRTOS::Semaphore("StopEvt"); +#endif - uint16_t m_numHandles; + /*************************************************************************** + * NimBLE private properties * + ***************************************************************************/ +#if defined(CONFIG_NIMBLE_ENABLED) + uint8_t m_removed; + ble_gatt_svc_def *m_pSvcDef; +#endif + + /*************************************************************************** + * Common private declarations * + ***************************************************************************/ + + BLEService(const char *uuid, uint16_t numHandles); + BLEService(BLEUUID uuid, uint16_t numHandles); + ~BLEService(); + + /*************************************************************************** + * Common private declarations * + ***************************************************************************/ + + void setHandle(uint16_t handle); + + /*************************************************************************** + * Bluedroid private declarations * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) BLECharacteristic *getLastCreatedCharacteristic(); void handleGATTServerEvent(esp_gatts_cb_event_t event, esp_gatt_if_t gatts_if, esp_ble_gatts_cb_param_t *param); - void setHandle(uint16_t handle); - //void setService(esp_gatt_srvc_id_t srvc_id); +#endif }; // BLEService -#endif /* CONFIG_BLUEDROID_ENABLED */ +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ #endif /* COMPONENTS_CPP_UTILS_BLESERVICE_H_ */ diff --git a/libraries/BLE/src/BLEServiceMap.cpp b/libraries/BLE/src/BLEServiceMap.cpp index 30a9db499f1..477da5b4cc2 100644 --- a/libraries/BLE/src/BLEServiceMap.cpp +++ b/libraries/BLE/src/BLEServiceMap.cpp @@ -3,16 +3,30 @@ * * Created on: Jun 22, 2017 * Author: kolban + * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on kolban's and h2zero's work) + * Description: Added support for NimBLE */ + #include "soc/soc_caps.h" #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) + +/***************************************************************************** + * Common includes * + *****************************************************************************/ + #include #include #include "BLEService.h" +/***************************************************************************** + * Common functions * + *****************************************************************************/ + /** * @brief Return the service by UUID. * @param [in] UUID The UUID to look up the service. @@ -82,13 +96,6 @@ String BLEServiceMap::toString() { return res; } // toString -void BLEServiceMap::handleGATTServerEvent(esp_gatts_cb_event_t event, esp_gatt_if_t gatts_if, esp_ble_gatts_cb_param_t *param) { - // Invoke the handler for every Service we have. - for (auto &myPair : m_uuidMap) { - myPair.first->handleGATTServerEvent(event, gatts_if, param); - } -} - /** * @brief Get the first service in the map. * @return The first service in the map. @@ -130,8 +137,21 @@ void BLEServiceMap::removeService(BLEService *service) { * @return amount of registered services */ int BLEServiceMap::getRegisteredServiceCount() { - return m_handleMap.size(); + return m_uuidMap.size(); +} + +/***************************************************************************** + * Bluedroid functions * + *****************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) +void BLEServiceMap::handleGATTServerEvent(esp_gatts_cb_event_t event, esp_gatt_if_t gatts_if, esp_ble_gatts_cb_param_t *param) { + // Invoke the handler for every Service we have. + for (auto &myPair : m_uuidMap) { + myPair.first->handleGATTServerEvent(event, gatts_if, param); + } } +#endif -#endif /* CONFIG_BLUEDROID_ENABLED */ +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ diff --git a/libraries/BLE/src/BLEUUID.cpp b/libraries/BLE/src/BLEUUID.cpp index 8074ea82f8f..d61fb695799 100644 --- a/libraries/BLE/src/BLEUUID.cpp +++ b/libraries/BLE/src/BLEUUID.cpp @@ -3,12 +3,22 @@ * * Created on: Jun 21, 2017 * Author: kolban + * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on kolban's and h2zero's work) + * Description: Added support for NimBLE */ + #include "soc/soc_caps.h" #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) + +/***************************************************************************** + * Common includes * + *****************************************************************************/ + #include #include #include @@ -18,25 +28,10 @@ #include "BLEUUID.h" #include "esp32-hal-log.h" -/** - * @brief Copy memory from source to target but in reverse order. - * - * When we move memory from one location it is normally: - * - * ``` - * [0][1][2]...[n] -> [0][1][2]...[n] - * ``` - * - * with this function, it is: - * - * ``` - * [0][1][2]...[n] -> [n][n-1][n-2]...[0] - * ``` - * - * @param [in] target The target of the copy - * @param [in] source The source of the copy - * @param [in] size The number of bytes to copy - */ +/***************************************************************************** + * Common functions * + *****************************************************************************/ + static void memrcpy(uint8_t *target, uint8_t *source, uint32_t size) { assert(size > 0); target += (size - 1); // Point target to the last byte of the target data @@ -48,29 +43,15 @@ static void memrcpy(uint8_t *target, uint8_t *source, uint32_t size) { } } // memrcpy -/** - * @brief Create a UUID from a string. - * - * Create a UUID from a string. There will be two possible stories here. Either the string represents - * a binary data field or the string represents a hex encoding of a UUID. - * For the hex encoding, here is an example: - * - * ``` - * "beb5483e-36e1-4688-b7f5-ea07361b26a8" - * 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 - * 12345678-90ab-cdef-1234-567890abcdef - * ``` - * - * This has a length of 36 characters. We need to parse this into 16 bytes. - * - * @param [in] value The string to build a UUID from. - */ +BLEUUID::BLEUUID() { + m_valueSet = false; +} // BLEUUID + BLEUUID::BLEUUID(String value) { - //Serial.printf("BLEUUID constructor from String=\"%s\"\n", value.c_str()); m_valueSet = true; if (value.length() == 4) { - m_uuid.len = ESP_UUID_LEN_16; - m_uuid.uuid.uuid16 = 0; + UUID_LEN(m_uuid) = BLE_UUID_16_BITS; + UUID_VAL_16(m_uuid) = 0; for (int i = 0; i < value.length();) { uint8_t MSB = value.c_str()[i]; uint8_t LSB = value.c_str()[i + 1]; @@ -81,12 +62,12 @@ BLEUUID::BLEUUID(String value) { if (LSB > '9') { LSB -= 7; } - m_uuid.uuid.uuid16 += (((MSB & 0x0F) << 4) | (LSB & 0x0F)) << (2 - i) * 4; + UUID_VAL_16(m_uuid) += (((MSB & 0x0F) << 4) | (LSB & 0x0F)) << (2 - i) * 4; i += 2; } } else if (value.length() == 8) { - m_uuid.len = ESP_UUID_LEN_32; - m_uuid.uuid.uuid32 = 0; + UUID_LEN(m_uuid) = BLE_UUID_32_BITS; + UUID_VAL_32(m_uuid) = 0; for (int i = 0; i < value.length();) { uint8_t MSB = value.c_str()[i]; uint8_t LSB = value.c_str()[i + 1]; @@ -97,18 +78,14 @@ BLEUUID::BLEUUID(String value) { if (LSB > '9') { LSB -= 7; } - m_uuid.uuid.uuid32 += (((MSB & 0x0F) << 4) | (LSB & 0x0F)) << (6 - i) * 4; + UUID_VAL_32(m_uuid) += (((MSB & 0x0F) << 4) | (LSB & 0x0F)) << (6 - i) * 4; i += 2; } - } else if (value.length() - == 16) { // How we can have 16 byte length string representing 128 bit uuid??? needs to be investigated (lack of time) - maybe raw data encoded as String (128b==16B)? - m_uuid.len = ESP_UUID_LEN_128; - memrcpy(m_uuid.uuid.uuid128, (uint8_t *)value.c_str(), 16); + } else if (value.length() == 16) { + UUID_LEN(m_uuid) = BLE_UUID_128_BITS; + memrcpy(UUID_VAL_128(m_uuid), (uint8_t *)value.c_str(), 16); } else if (value.length() == 36) { - //log_d("36 characters:"); - // If the length of the string is 36 bytes then we will assume it is a long hex string in - // UUID format. - m_uuid.len = ESP_UUID_LEN_128; + UUID_LEN(m_uuid) = BLE_UUID_128_BITS; int n = 0; for (int i = 0; i < value.length();) { if (value.c_str()[i] == '-') { @@ -123,7 +100,7 @@ BLEUUID::BLEUUID(String value) { if (LSB > '9') { LSB -= 7; } - m_uuid.uuid.uuid128[15 - n++] = ((MSB & 0x0F) << 4) | (LSB & 0x0F); + UUID_VAL_128(m_uuid)[15 - n++] = ((MSB & 0x0F) << 4) | (LSB & 0x0F); i += 2; } } else { @@ -132,128 +109,73 @@ BLEUUID::BLEUUID(String value) { } } //BLEUUID(String) -/* -BLEUUID::BLEUUID(String value) { - this.BLEUUID(String(value.c_str(), value.length())); -} //BLEUUID(String) -*/ - -/** - * @brief Create a UUID from 16 bytes of memory. - * - * @param [in] pData The pointer to the start of the UUID. - * @param [in] size The size of the data. - * @param [in] msbFirst Is the MSB first in pData memory? - */ BLEUUID::BLEUUID(uint8_t *pData, size_t size, bool msbFirst) { if (size != 16) { log_e("ERROR: UUID length not 16 bytes"); return; } - m_uuid.len = ESP_UUID_LEN_128; + UUID_LEN(m_uuid) = BLE_UUID_128_BITS; if (msbFirst) { - memrcpy(m_uuid.uuid.uuid128, pData, 16); + memrcpy(UUID_VAL_128(m_uuid), pData, 16); } else { - memcpy(m_uuid.uuid.uuid128, pData, 16); + memcpy(UUID_VAL_128(m_uuid), pData, 16); } m_valueSet = true; } // BLEUUID -/** - * @brief Create a UUID from the 16bit value. - * - * @param [in] uuid The 16bit short form UUID. - */ BLEUUID::BLEUUID(uint16_t uuid) { - m_uuid.len = ESP_UUID_LEN_16; - m_uuid.uuid.uuid16 = uuid; + UUID_LEN(m_uuid) = BLE_UUID_16_BITS; + UUID_VAL_16(m_uuid) = uuid; m_valueSet = true; } // BLEUUID -/** - * @brief Create a UUID from the 32bit value. - * - * @param [in] uuid The 32bit short form UUID. - */ BLEUUID::BLEUUID(uint32_t uuid) { - m_uuid.len = ESP_UUID_LEN_32; - m_uuid.uuid.uuid32 = uuid; + UUID_LEN(m_uuid) = BLE_UUID_32_BITS; + UUID_VAL_32(m_uuid) = uuid; m_valueSet = true; } // BLEUUID -/** - * @brief Create a UUID from the native UUID. - * - * @param [in] uuid The native UUID. - */ -BLEUUID::BLEUUID(esp_bt_uuid_t uuid) { - m_uuid = uuid; +BLEUUID::BLEUUID(uint32_t first, uint16_t second, uint16_t third, uint64_t fourth) { + UUID_LEN(m_uuid) = BLE_UUID_128_BITS; + memcpy(UUID_VAL_128(m_uuid) + 12, &first, 4); + memcpy(UUID_VAL_128(m_uuid) + 10, &second, 2); + memcpy(UUID_VAL_128(m_uuid) + 8, &third, 2); + memcpy(UUID_VAL_128(m_uuid), &fourth, 8); m_valueSet = true; -} // BLEUUID - -/** - * @brief Create a UUID from the ESP32 esp_gat_id_t. - * - * @param [in] gattId The data to create the UUID from. - */ -BLEUUID::BLEUUID(esp_gatt_id_t gattId) : BLEUUID(gattId.uuid) {} // BLEUUID - -BLEUUID::BLEUUID() { - m_valueSet = false; -} // BLEUUID +} -/** - * @brief Get the number of bits in this uuid. - * @return The number of bits in the UUID. One of 16, 32 or 128. - */ uint8_t BLEUUID::bitSize() { if (!m_valueSet) { return 0; } - switch (m_uuid.len) { - case ESP_UUID_LEN_16: return 16; - case ESP_UUID_LEN_32: return 32; - case ESP_UUID_LEN_128: return 128; - default: log_e("Unknown UUID length: %d", m_uuid.len); return 0; + switch (UUID_LEN(m_uuid)) { + case BLE_UUID_16_BITS: return 16; + case BLE_UUID_32_BITS: return 32; + case BLE_UUID_128_BITS: return 128; + default: log_e("Unknown UUID length: %d", UUID_LEN(m_uuid)); return 0; } // End of switch } // bitSize -/** - * @brief Compare a UUID against this UUID. - * - * @param [in] uuid The UUID to compare against. - * @return True if the UUIDs are equal and false otherwise. - */ -bool BLEUUID::equals(BLEUUID uuid) { - //log_d("Comparing: %s to %s", toString().c_str(), uuid.toString().c_str()); +bool BLEUUID::equals(const BLEUUID &uuid) const { if (!m_valueSet || !uuid.m_valueSet) { return false; } - if (uuid.m_uuid.len != m_uuid.len) { + if (UUID_LEN(uuid.m_uuid) != UUID_LEN(m_uuid)) { return uuid.toString() == toString(); } - if (uuid.m_uuid.len == ESP_UUID_LEN_16) { - return uuid.m_uuid.uuid.uuid16 == m_uuid.uuid.uuid16; + if (UUID_LEN(uuid.m_uuid) == BLE_UUID_16_BITS) { + return UUID_VAL_16(uuid.m_uuid) == UUID_VAL_16(m_uuid); } - if (uuid.m_uuid.len == ESP_UUID_LEN_32) { - return uuid.m_uuid.uuid.uuid32 == m_uuid.uuid.uuid32; + if (UUID_LEN(uuid.m_uuid) == BLE_UUID_32_BITS) { + return UUID_VAL_32(uuid.m_uuid) == UUID_VAL_32(m_uuid); } - return memcmp(uuid.m_uuid.uuid.uuid128, m_uuid.uuid.uuid128, 16) == 0; + return memcmp(UUID_VAL_128(uuid.m_uuid), UUID_VAL_128(m_uuid), 16) == 0; } // equals -/** - * Create a BLEUUID from a string of the form: - * 0xNNNN - * 0xNNNNNNNN - * 0x - * NNNN - * NNNNNNNN - * - */ BLEUUID BLEUUID::fromString(String _uuid) { uint8_t start = 0; if (strstr(_uuid.c_str(), "0x") != nullptr) { // If the string starts with 0x, skip those characters. @@ -273,117 +195,142 @@ BLEUUID BLEUUID::fromString(String _uuid) { return BLEUUID(); } // fromString -/** - * @brief Get the native UUID value. - * - * @return The native UUID value or NULL if not set. - */ -esp_bt_uuid_t *BLEUUID::getNative() { - //log_d(">> getNative()") - if (m_valueSet == false) { - log_v("<< Return of un-initialized UUID!"); - return nullptr; - } - //log_d("<< getNative()"); - return &m_uuid; -} // getNative - -/** - * @brief Convert a UUID to its 128 bit representation. - * - * A UUID can be internally represented as 16bit, 32bit or the full 128bit. This method - * will convert 16 or 32 bit representations to the full 128bit. - */ BLEUUID BLEUUID::to128() { - //log_v(">> toFull() - %s", toString().c_str()); - - // If we either don't have a value or are already a 128 bit UUID, nothing further to do. - if (!m_valueSet || m_uuid.len == ESP_UUID_LEN_128) { + if (!m_valueSet || UUID_LEN(m_uuid) == BLE_UUID_128_BITS) { return *this; } - // If we are 16 bit or 32 bit, then set the 4 bytes of the variable part of the UUID. - if (m_uuid.len == ESP_UUID_LEN_16) { - uint16_t temp = m_uuid.uuid.uuid16; - m_uuid.uuid.uuid128[15] = 0; - m_uuid.uuid.uuid128[14] = 0; - m_uuid.uuid.uuid128[13] = (temp >> 8) & 0xff; - m_uuid.uuid.uuid128[12] = temp & 0xff; - - } else if (m_uuid.len == ESP_UUID_LEN_32) { - uint32_t temp = m_uuid.uuid.uuid32; - m_uuid.uuid.uuid128[15] = (temp >> 24) & 0xff; - m_uuid.uuid.uuid128[14] = (temp >> 16) & 0xff; - m_uuid.uuid.uuid128[13] = (temp >> 8) & 0xff; - m_uuid.uuid.uuid128[12] = temp & 0xff; + if (UUID_LEN(m_uuid) == BLE_UUID_16_BITS) { + uint16_t temp = UUID_VAL_16(m_uuid); + UUID_VAL_128(m_uuid)[15] = 0; + UUID_VAL_128(m_uuid)[14] = 0; + UUID_VAL_128(m_uuid)[13] = (temp >> 8) & 0xff; + UUID_VAL_128(m_uuid)[12] = temp & 0xff; + } else if (UUID_LEN(m_uuid) == BLE_UUID_32_BITS) { + uint16_t temp = UUID_VAL_32(m_uuid); + UUID_VAL_128(m_uuid)[15] = (temp >> 24) & 0xff; + UUID_VAL_128(m_uuid)[14] = (temp >> 16) & 0xff; + UUID_VAL_128(m_uuid)[13] = (temp >> 8) & 0xff; + UUID_VAL_128(m_uuid)[12] = temp & 0xff; } - // Set the fixed parts of the UUID. - m_uuid.uuid.uuid128[11] = 0x00; - m_uuid.uuid.uuid128[10] = 0x00; - - m_uuid.uuid.uuid128[9] = 0x10; - m_uuid.uuid.uuid128[8] = 0x00; + UUID_VAL_128(m_uuid)[11] = 0x00; + UUID_VAL_128(m_uuid)[10] = 0x00; + UUID_VAL_128(m_uuid)[9] = 0x10; + UUID_VAL_128(m_uuid)[8] = 0x00; + UUID_VAL_128(m_uuid)[7] = 0x80; + UUID_VAL_128(m_uuid)[6] = 0x00; + UUID_VAL_128(m_uuid)[5] = 0x00; + UUID_VAL_128(m_uuid)[4] = 0x80; + UUID_VAL_128(m_uuid)[3] = 0x5f; + UUID_VAL_128(m_uuid)[2] = 0x9b; + UUID_VAL_128(m_uuid)[1] = 0x34; + UUID_VAL_128(m_uuid)[0] = 0xfb; + + UUID_LEN(m_uuid) = BLE_UUID_128_BITS; + return *this; +} // to128 - m_uuid.uuid.uuid128[7] = 0x80; - m_uuid.uuid.uuid128[6] = 0x00; +BLEUUID BLEUUID::to16() { + if (!m_valueSet || UUID_LEN(m_uuid) == BLE_UUID_16_BITS) { + return *this; + } - m_uuid.uuid.uuid128[5] = 0x00; - m_uuid.uuid.uuid128[4] = 0x80; - m_uuid.uuid.uuid128[3] = 0x5f; - m_uuid.uuid.uuid128[2] = 0x9b; - m_uuid.uuid.uuid128[1] = 0x34; - m_uuid.uuid.uuid128[0] = 0xfb; + if (UUID_LEN(m_uuid) == BLE_UUID_128_BITS) { + uint8_t base128[] = {0xfb, 0x34, 0x9b, 0x5f, 0x80, 0x00, 0x00, 0x80, 0x00, 0x10, 0x00, 0x00}; + if (memcmp(UUID_VAL_128(m_uuid), base128, sizeof(base128)) == 0) { + *this = BLEUUID(*(uint16_t *)(UUID_VAL_128(m_uuid) + 12)); + } + } - m_uuid.len = ESP_UUID_LEN_128; - //log_d("<< toFull <- %s", toString().c_str()); return *this; -} // to128 +} -/** - * @brief Get a string representation of the UUID. - * - * The format of a string is: - * 01234567 8901 2345 6789 012345678901 - * 0000180d-0000-1000-8000-00805f9b34fb - * 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 - * - * @return A string representation of the UUID. - */ -String BLEUUID::toString() { +String BLEUUID::toString() const { if (!m_valueSet) { return ""; // If we have no value, nothing to format. } - // If the UUIDs are 16 or 32 bit, pad correctly. - if (m_uuid.len == ESP_UUID_LEN_16) { // If the UUID is 16bit, pad correctly. + if (UUID_LEN(m_uuid) == BLE_UUID_16_BITS) { // If the UUID is 16bit, pad correctly. char hex[9]; - snprintf(hex, sizeof(hex), "%08x", m_uuid.uuid.uuid16); + snprintf(hex, sizeof(hex), "%08x", UUID_VAL_16(m_uuid)); return String(hex) + "-0000-1000-8000-00805f9b34fb"; } // End 16bit UUID - if (m_uuid.len == ESP_UUID_LEN_32) { // If the UUID is 32bit, pad correctly. + if (UUID_LEN(m_uuid) == BLE_UUID_32_BITS) { // If the UUID is 32bit, pad correctly. char hex[9]; - snprintf(hex, sizeof(hex), "%08lx", m_uuid.uuid.uuid32); + snprintf(hex, sizeof(hex), "%08lx", UUID_VAL_32(m_uuid)); return String(hex) + "-0000-1000-8000-00805f9b34fb"; } // End 32bit UUID // The UUID is not 16bit or 32bit which means that it is 128bit. - // - // UUID string format: - // AABBCCDD-EEFF-GGHH-IIJJ-KKLLMMNNOOPP auto size = 37; // 32 for UUID data, 4 for '-' delimiters and one for a terminator == 37 chars char *hex = (char *)malloc(size); snprintf( - hex, size, "%02x%02x%02x%02x-%02x%02x-%02x%02x-%02x%02x-%02x%02x%02x%02x%02x%02x", m_uuid.uuid.uuid128[15], m_uuid.uuid.uuid128[14], - m_uuid.uuid.uuid128[13], m_uuid.uuid.uuid128[12], m_uuid.uuid.uuid128[11], m_uuid.uuid.uuid128[10], m_uuid.uuid.uuid128[9], m_uuid.uuid.uuid128[8], - m_uuid.uuid.uuid128[7], m_uuid.uuid.uuid128[6], m_uuid.uuid.uuid128[5], m_uuid.uuid.uuid128[4], m_uuid.uuid.uuid128[3], m_uuid.uuid.uuid128[2], - m_uuid.uuid.uuid128[1], m_uuid.uuid.uuid128[0] + hex, size, "%02x%02x%02x%02x-%02x%02x-%02x%02x-%02x%02x-%02x%02x%02x%02x%02x%02x", UUID_VAL_128(m_uuid)[15], UUID_VAL_128(m_uuid)[14], + UUID_VAL_128(m_uuid)[13], UUID_VAL_128(m_uuid)[12], UUID_VAL_128(m_uuid)[11], UUID_VAL_128(m_uuid)[10], UUID_VAL_128(m_uuid)[9], UUID_VAL_128(m_uuid)[8], + UUID_VAL_128(m_uuid)[7], UUID_VAL_128(m_uuid)[6], UUID_VAL_128(m_uuid)[5], UUID_VAL_128(m_uuid)[4], UUID_VAL_128(m_uuid)[3], UUID_VAL_128(m_uuid)[2], + UUID_VAL_128(m_uuid)[1], UUID_VAL_128(m_uuid)[0] ); + String res(hex); free(hex); return res; } // toString -#endif /* CONFIG_BLUEDROID_ENABLED */ +bool BLEUUID::operator==(const BLEUUID &rhs) const { + return equals(rhs); +} + +bool BLEUUID::operator!=(const BLEUUID &rhs) const { + return !equals(rhs); +} + +/***************************************************************************** + * Bluedroid functions * + *****************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) +BLEUUID::BLEUUID(esp_bt_uuid_t uuid) { + m_uuid = uuid; + m_valueSet = true; +} // BLEUUID + +BLEUUID::BLEUUID(esp_gatt_id_t gattId) : BLEUUID(gattId.uuid) {} // BLEUUID + +esp_bt_uuid_t *BLEUUID::getNative() { + if (m_valueSet == false) { + log_v("<< Return of un-initialized UUID!"); + return nullptr; + } + return &m_uuid; +} // getNative +#endif + +/***************************************************************************** + * NimBLE functions * + *****************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) +BLEUUID::BLEUUID(ble_uuid_any_t uuid) { + m_uuid = uuid; + m_valueSet = true; +} // BLEUUID + +BLEUUID::BLEUUID(const ble_uuid128_t *uuid) { + m_uuid.u.type = BLE_UUID_TYPE_128; + memcpy(m_uuid.u128.value, uuid->value, 16); + m_valueSet = true; +} // BLEUUID + +const ble_uuid_any_t *BLEUUID::getNative() const { + if (m_valueSet == false) { + log_v("<< Return of un-initialized UUID!"); + return nullptr; + } + return &m_uuid; +} // getNative +#endif + +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ diff --git a/libraries/BLE/src/BLEUUID.h b/libraries/BLE/src/BLEUUID.h index 1be013942e3..43c4e91b01d 100644 --- a/libraries/BLE/src/BLEUUID.h +++ b/libraries/BLE/src/BLEUUID.h @@ -3,42 +3,125 @@ * * Created on: Jun 21, 2017 * Author: kolban + * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on kolban's and h2zero's work) + * Description: Added support for NimBLE */ #ifndef COMPONENTS_CPP_UTILS_BLEUUID_H_ #define COMPONENTS_CPP_UTILS_BLEUUID_H_ #include "soc/soc_caps.h" -#include "WString.h" #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if CONFIG_BLUEDROID_ENABLED +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) + +/***************************************************************************** + * Common includes * + *****************************************************************************/ + +#include "WString.h" + +/***************************************************************************** + * Bluedroid includes and definitions * + *****************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) #include +#define BLE_UUID_16_BITS ESP_UUID_LEN_16 +#define BLE_UUID_32_BITS ESP_UUID_LEN_32 +#define BLE_UUID_128_BITS ESP_UUID_LEN_128 +#define UUID_LEN(s) s.len +#define UUID_VAL_16(s) s.uuid.uuid16 +#define UUID_VAL_32(s) s.uuid.uuid32 +#define UUID_VAL_128(s) s.uuid.uuid128 +#endif + +/***************************************************************************** + * NimBLE includes and definitions * + *****************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) +#include +#define BLE_UUID_16_BITS BLE_UUID_TYPE_16 +#define BLE_UUID_32_BITS BLE_UUID_TYPE_32 +#define BLE_UUID_128_BITS BLE_UUID_TYPE_128 +#define UUID_LEN(s) s.u.type +#define UUID_VAL_16(s) s.u16.value +#define UUID_VAL_32(s) s.u32.value +#define UUID_VAL_128(s) s.u128.value +#endif /** * @brief A model of a %BLE UUID. */ class BLEUUID { public: + /*************************************************************************** + * Common public declarations * + ***************************************************************************/ + BLEUUID(String uuid); BLEUUID(uint16_t uuid); BLEUUID(uint32_t uuid); - BLEUUID(esp_bt_uuid_t uuid); BLEUUID(uint8_t *pData, size_t size, bool msbFirst); - BLEUUID(esp_gatt_id_t gattId); + BLEUUID(uint32_t first, uint16_t second, uint16_t third, uint64_t fourth); BLEUUID(); uint8_t bitSize(); // Get the number of bits in this uuid. - bool equals(BLEUUID uuid); - esp_bt_uuid_t *getNative(); + bool equals(const BLEUUID &uuid) const; BLEUUID to128(); - String toString(); + BLEUUID to16(); + String toString() const; static BLEUUID fromString(String uuid); // Create a BLEUUID from a string + bool operator==(const BLEUUID &rhs) const; + bool operator!=(const BLEUUID &rhs) const; + + /*************************************************************************** + * Bluedroid public declarations * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) + BLEUUID(esp_bt_uuid_t uuid); + BLEUUID(esp_gatt_id_t gattId); + esp_bt_uuid_t *getNative(); +#endif + + /*************************************************************************** + * NimBLE public declarations * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) + BLEUUID(ble_uuid_any_t uuid); + BLEUUID(const ble_uuid128_t *uuid); + const ble_uuid_any_t *getNative() const; +#endif + private: - esp_bt_uuid_t m_uuid; // The underlying UUID structure that this class wraps. + /*************************************************************************** + * Common private properties * + ***************************************************************************/ + bool m_valueSet = false; // Is there a value set for this instance. + + /*************************************************************************** + * Bluedroid private properties * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) + esp_bt_uuid_t m_uuid; // The underlying UUID structure that this class wraps. +#endif + + /*************************************************************************** + * NimBLE private properties * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) + ble_uuid_any_t m_uuid; // The underlying UUID structure that this class wraps. +#endif }; // BLEUUID -#endif /* CONFIG_BLUEDROID_ENABLED */ +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ #endif /* COMPONENTS_CPP_UTILS_BLEUUID_H_ */ diff --git a/libraries/BLE/src/BLEUtils.cpp b/libraries/BLE/src/BLEUtils.cpp index 05e1e32deed..4ca04e6e2b6 100644 --- a/libraries/BLE/src/BLEUtils.cpp +++ b/libraries/BLE/src/BLEUtils.cpp @@ -3,12 +3,22 @@ * * Created on: Mar 25, 2017 * Author: kolban + * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on kolban's and h2zero's work) + * Description: Added support for NimBLE */ + #include "soc/soc_caps.h" #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) + +/***************************************************************************** + * Common includes * + *****************************************************************************/ + #include "BLEAddress.h" #include "BLEClient.h" #include "BLEUtils.h" @@ -17,17 +27,56 @@ #include #include -#include // ESP32 BLE -#include // ESP32 BLE -#include // ESP32 BLE -#include // ESP32 BLE -#include // ESP32 ESP-IDF -#include // Part of C++ STL +#include + +#include +#include #include #include #include "esp32-hal-log.h" +/***************************************************************************** + * Bluedroid includes * + *****************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) +#include +#include +#include +#endif + +/***************************************************************************** + * NimBLE includes and definitions * + *****************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) +#include +#include +#include +#include +#include +#include +#include + +#if ARDUHAL_LOG_LEVEL >= ARDUHAL_LOG_LEVEL_DEBUG +#define CONFIG_NIMBLE_ENABLE_RETURN_CODE_TEXT +#define CONFIG_NIMBLE_ENABLE_ADVERTISMENT_TYPE_TEXT +#define CONFIG_NIMBLE_ENABLE_GAP_EVENT_CODE_TEXT +#endif + +#ifndef CONFIG_NIMBLE_FREERTOS_TASK_BLOCK_BIT +#define CONFIG_NIMBLE_FREERTOS_TASK_BLOCK_BIT 31 +#endif + +constexpr uint32_t TASK_BLOCK_BIT = (1 << CONFIG_NIMBLE_FREERTOS_TASK_BLOCK_BIT); +#endif + +/***************************************************************************** + * Bluedroid types and constants * + *****************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) typedef struct { uint32_t assignedNumber; const char *name; @@ -593,90 +642,58 @@ static const gattService_t g_gattServices[] = { #endif {"", "", 0} }; +#endif -/** - * @brief Convert characteristic properties into a string representation. - * @param [in] prop Characteristic properties. - * @return A string representation of characteristic properties. - */ -String BLEUtils::characteristicPropertiesToString(esp_gatt_char_prop_t prop) { - String res = "broadcast: "; - res += ((prop & ESP_GATT_CHAR_PROP_BIT_BROADCAST) ? "1" : "0"); - res += ", read: "; - res += ((prop & ESP_GATT_CHAR_PROP_BIT_READ) ? "1" : "0"); - res += ", write_nr: "; - res += ((prop & ESP_GATT_CHAR_PROP_BIT_WRITE_NR) ? "1" : "0"); - res += ", write: "; - res += ((prop & ESP_GATT_CHAR_PROP_BIT_WRITE) ? "1" : "0"); - res += ", notify: "; - res += ((prop & ESP_GATT_CHAR_PROP_BIT_NOTIFY) ? "1" : "0"); - res += ", indicate: "; - res += ((prop & ESP_GATT_CHAR_PROP_BIT_INDICATE) ? "1" : "0"); - res += ", auth: "; - res += ((prop & ESP_GATT_CHAR_PROP_BIT_AUTH) ? "1" : "0"); - return res; -} // characteristicPropertiesToString - -/** - * @brief Convert an esp_gatt_id_t to a string. - */ -static String gattIdToString(esp_gatt_id_t gattId) { - String res = "uuid: " + BLEUUID(gattId.uuid).toString() + ", inst_id: "; - char val[8]; - snprintf(val, sizeof(val), "%d", (int)gattId.inst_id); - res += val; - return res; -} // gattIdToString +/***************************************************************************** + * Common functions * + *****************************************************************************/ /** - * @brief Convert an esp_ble_addr_type_t to a string representation. + * @brief Create a hex representation of data. + * + * @param [in] target Where to write the hex string. If this is null, we malloc storage. + * @param [in] source The start of the binary data. + * @param [in] length The length of the data to convert. + * @return A pointer to the formatted buffer. */ -const char *BLEUtils::addressTypeToString(esp_ble_addr_type_t type) { - switch (type) { -#if ARDUHAL_LOG_LEVEL >= ARDUHAL_LOG_LEVEL_DEBUG - case BLE_ADDR_TYPE_PUBLIC: return "BLE_ADDR_TYPE_PUBLIC"; - case BLE_ADDR_TYPE_RANDOM: return "BLE_ADDR_TYPE_RANDOM"; - case BLE_ADDR_TYPE_RPA_PUBLIC: return "BLE_ADDR_TYPE_RPA_PUBLIC"; - case BLE_ADDR_TYPE_RPA_RANDOM: return "BLE_ADDR_TYPE_RPA_RANDOM"; -#endif - default: return " esp_ble_addr_type_t"; +char *BLEUtils::buildHexData(uint8_t *target, uint8_t *source, uint8_t length) { + // Guard against too much data. + if (length > 100) { + length = 100; } -} // addressTypeToString -/** - * @brief Convert the BLE Advertising Data flags to a string. - * @param adFlags The flags to convert - * @return String A string representation of the advertising flags. - */ -String BLEUtils::adFlagsToString(uint8_t adFlags) { - String res; - if (adFlags & (1 << 0)) { - res += "[LE Limited Discoverable Mode] "; - } - if (adFlags & (1 << 1)) { - res += "[LE General Discoverable Mode] "; - } - if (adFlags & (1 << 2)) { - res += "[BR/EDR Not Supported] "; + if (target == nullptr) { + target = (uint8_t *)malloc(length * 2 + 1); + if (target == nullptr) { + log_e("buildHexData: malloc failed"); + return nullptr; + } } - if (adFlags & (1 << 3)) { - res += "[Simultaneous LE and BR/EDR to Same Device Capable (Controller)] "; + char *startOfData = (char *)target; + + for (int i = 0; i < length; i++) { + sprintf((char *)target, "%.2x", (char)*source); + source++; + target += 2; } - if (adFlags & (1 << 4)) { - res += "[Simultaneous LE and BR/EDR to Same Device Capable (Host)] "; + + // Handle the special case where there was no data. + if (length == 0) { + *startOfData = 0; } - return res; -} // adFlagsToString + + return startOfData; +} // buildHexData /** - * @brief Given an advertising type, return a string representation of the type. + * @brief Given an advertising data type, return a string representation of the type. * * For details see ... * https://www.bluetooth.com/specifications/assigned-numbers/generic-access-profile * * @return A string representation of the type. */ -const char *BLEUtils::advTypeToString(uint8_t advType) { +const char *BLEUtils::advDataTypeToString(uint8_t advType) { switch (advType) { #if ARDUHAL_LOG_LEVEL >= ARDUHAL_LOG_LEVEL_DEBUG case ESP_BLE_AD_TYPE_FLAG: // 0x01 @@ -721,7 +738,8 @@ const char *BLEUtils::advTypeToString(uint8_t advType) { return "ESP_BLE_AD_TYPE_APPEARANCE"; case ESP_BLE_AD_TYPE_ADV_INT: // 0x1a return "ESP_BLE_AD_TYPE_ADV_INT"; - case ESP_BLE_AD_TYPE_32SOL_SRV_UUID: return "ESP_BLE_AD_TYPE_32SOL_SRV_UUID"; + case ESP_BLE_AD_TYPE_32SOL_SRV_UUID: // 0x1f + return "ESP_BLE_AD_TYPE_32SOL_SRV_UUID"; case ESP_BLE_AD_TYPE_32SERVICE_DATA: // 0x20 return "ESP_BLE_AD_TYPE_32SERVICE_DATA"; case ESP_BLE_AD_TYPE_128SERVICE_DATA: // 0x21 @@ -731,58 +749,36 @@ const char *BLEUtils::advTypeToString(uint8_t advType) { #endif default: log_v(" adv data type: 0x%x", advType); return ""; } // End switch -} // advTypeToString +} // advDataTypeToString -esp_gatt_id_t BLEUtils::buildGattId(esp_bt_uuid_t uuid, uint8_t inst_id) { - esp_gatt_id_t retGattId; - retGattId.uuid = uuid; - retGattId.inst_id = inst_id; - return retGattId; -} +/***************************************************************************** + * Bluedroid functions * + *****************************************************************************/ -esp_gatt_srvc_id_t BLEUtils::buildGattSrvcId(esp_gatt_id_t gattId, bool is_primary) { - esp_gatt_srvc_id_t retSrvcId; - retSrvcId.id = gattId; - retSrvcId.is_primary = is_primary; - return retSrvcId; -} +#if defined(CONFIG_BLUEDROID_ENABLED) /** - * @brief Create a hex representation of data. - * - * @param [in] target Where to write the hex string. If this is null, we malloc storage. - * @param [in] source The start of the binary data. - * @param [in] length The length of the data to convert. - * @return A pointer to the formatted buffer. + * @brief Convert characteristic properties into a string representation. + * @param [in] prop Characteristic properties. + * @return A string representation of characteristic properties. */ -char *BLEUtils::buildHexData(uint8_t *target, uint8_t *source, uint8_t length) { - // Guard against too much data. - if (length > 100) { - length = 100; - } - - if (target == nullptr) { - target = (uint8_t *)malloc(length * 2 + 1); - if (target == nullptr) { - log_e("buildHexData: malloc failed"); - return nullptr; - } - } - char *startOfData = (char *)target; - - for (int i = 0; i < length; i++) { - sprintf((char *)target, "%.2x", (char)*source); - source++; - target += 2; - } - - // Handle the special case where there was no data. - if (length == 0) { - *startOfData = 0; - } - - return startOfData; -} // buildHexData +String BLEUtils::characteristicPropertiesToString(uint8_t prop) { + String res = "broadcast: "; + res += ((prop & ESP_GATT_CHAR_PROP_BIT_BROADCAST) ? "1" : "0"); + res += ", read: "; + res += ((prop & ESP_GATT_CHAR_PROP_BIT_READ) ? "1" : "0"); + res += ", write_nr: "; + res += ((prop & ESP_GATT_CHAR_PROP_BIT_WRITE_NR) ? "1" : "0"); + res += ", write: "; + res += ((prop & ESP_GATT_CHAR_PROP_BIT_WRITE) ? "1" : "0"); + res += ", notify: "; + res += ((prop & ESP_GATT_CHAR_PROP_BIT_NOTIFY) ? "1" : "0"); + res += ", indicate: "; + res += ((prop & ESP_GATT_CHAR_PROP_BIT_INDICATE) ? "1" : "0"); + res += ", auth: "; + res += ((prop & ESP_GATT_CHAR_PROP_BIT_AUTH) ? "1" : "0"); + return res; +} // characteristicPropertiesToString /** * @brief Build a printable string of memory range. @@ -802,6 +798,71 @@ String BLEUtils::buildPrintData(uint8_t *source, size_t length) { return res; } // buildPrintData +/** + * @brief Convert an esp_gatt_id_t to a string. + */ +static String gattIdToString(esp_gatt_id_t gattId) { + String res = "uuid: " + BLEUUID(gattId.uuid).toString() + ", inst_id: "; + char val[8]; + snprintf(val, sizeof(val), "%d", (int)gattId.inst_id); + res += val; + return res; +} // gattIdToString + +/** + * @brief Convert an esp_ble_addr_type_t to a string representation. + */ +const char *BLEUtils::addressTypeToString(esp_ble_addr_type_t type) { + switch (type) { +#if ARDUHAL_LOG_LEVEL >= ARDUHAL_LOG_LEVEL_DEBUG + case BLE_ADDR_TYPE_PUBLIC: return "BLE_ADDR_TYPE_PUBLIC"; + case BLE_ADDR_TYPE_RANDOM: return "BLE_ADDR_TYPE_RANDOM"; + case BLE_ADDR_TYPE_RPA_PUBLIC: return "BLE_ADDR_TYPE_RPA_PUBLIC"; + case BLE_ADDR_TYPE_RPA_RANDOM: return "BLE_ADDR_TYPE_RPA_RANDOM"; +#endif + default: return " esp_ble_addr_type_t"; + } +} // addressTypeToString + +/** + * @brief Convert the BLE Advertising Data flags to a string. + * @param adFlags The flags to convert + * @return String A string representation of the advertising flags. + */ +String BLEUtils::adFlagsToString(uint8_t adFlags) { + String res; + if (adFlags & (1 << 0)) { + res += "[LE Limited Discoverable Mode] "; + } + if (adFlags & (1 << 1)) { + res += "[LE General Discoverable Mode] "; + } + if (adFlags & (1 << 2)) { + res += "[BR/EDR Not Supported] "; + } + if (adFlags & (1 << 3)) { + res += "[Simultaneous LE and BR/EDR to Same Device Capable (Controller)] "; + } + if (adFlags & (1 << 4)) { + res += "[Simultaneous LE and BR/EDR to Same Device Capable (Host)] "; + } + return res; +} // adFlagsToString + +esp_gatt_id_t BLEUtils::buildGattId(esp_bt_uuid_t uuid, uint8_t inst_id) { + esp_gatt_id_t retGattId; + retGattId.uuid = uuid; + retGattId.inst_id = inst_id; + return retGattId; +} + +esp_gatt_srvc_id_t BLEUtils::buildGattSrvcId(esp_gatt_id_t gattId, bool is_primary) { + esp_gatt_srvc_id_t retSrvcId; + retSrvcId.id = gattId; + retSrvcId.is_primary = is_primary; + return retSrvcId; +} + /** * @brief Convert a close/disconnect reason to a string. * @param [in] reason The close reason. @@ -1817,5 +1878,380 @@ const char *BLEUtils::searchEventTypeToString(esp_gap_search_evt_t searchEvt) { } } // searchEventTypeToString -#endif /* CONFIG_BLUEDROID_ENABLED */ +#endif // CONFIG_BLUEDROID_ENABLED + +/***************************************************************************** + * NimBLE functions * + *****************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) + +/** + * @brief Construct a BLETaskData instance. + * @param [in] pInstance An instance of the class that will be waiting. + * @param [in] flags General purpose flags for the caller. + * @param [in] buf A buffer for data. + */ +BLETaskData::BLETaskData(void *pInstance, int flags, void *buf) : m_pInstance{pInstance}, m_flags{flags}, m_pBuf{buf}, m_pHandle{xTaskGetCurrentTaskHandle()} {} + +/** + * @brief Destructor. + */ +BLETaskData::~BLETaskData() {} + +/** + * @brief A function for checking validity of connection parameters. + * @param [in] params A pointer to the structure containing the parameters to check. + * @return valid == 0 or error code. + */ +int BLEUtils::checkConnParams(ble_gap_conn_params *params) { + /* Check connection interval min */ + if ((params->itvl_min < BLE_HCI_CONN_ITVL_MIN) || (params->itvl_min > BLE_HCI_CONN_ITVL_MAX)) { + return BLE_ERR_INV_HCI_CMD_PARMS; + } + /* Check connection interval max */ + if ((params->itvl_max < BLE_HCI_CONN_ITVL_MIN) || (params->itvl_max > BLE_HCI_CONN_ITVL_MAX) || (params->itvl_max < params->itvl_min)) { + return BLE_ERR_INV_HCI_CMD_PARMS; + } + + /* Check connection latency */ + if (params->latency > BLE_HCI_CONN_LATENCY_MAX) { + return BLE_ERR_INV_HCI_CMD_PARMS; + } + + /* Check supervision timeout */ + if ((params->supervision_timeout < BLE_HCI_CONN_SPVN_TIMEOUT_MIN) || (params->supervision_timeout > BLE_HCI_CONN_SPVN_TIMEOUT_MAX)) { + return BLE_ERR_INV_HCI_CMD_PARMS; + } + + /* Check connection event length */ + if (params->min_ce_len > params->max_ce_len) { + return BLE_ERR_INV_HCI_CMD_PARMS; + } + + return 0; +} + +/** + * @brief Converts a return code from the NimBLE stack to a text string. + * @param [in] rc The return code to convert. + * @return A string representation of the return code. + */ +const char *BLEUtils::returnCodeToString(int rc) { +#if defined(CONFIG_NIMBLE_ENABLE_RETURN_CODE_TEXT) + switch (rc) { + case 0: return "SUCCESS"; + case BLE_HS_EAGAIN: return "Temporary failure; try again."; + case BLE_HS_EALREADY: return "Operation already in progress or completed."; + case BLE_HS_EINVAL: return "One or more arguments are invalid."; + case BLE_HS_EMSGSIZE: return "The provided buffer is too small."; + case BLE_HS_ENOENT: return "No entry matching the specified criteria."; + case BLE_HS_ENOMEM: return "Operation failed due to resource exhaustion."; + case BLE_HS_ENOTCONN: return "No open connection with the specified handle."; + case BLE_HS_ENOTSUP: return "Operation disabled at compile time."; + case BLE_HS_EAPP: return "Application callback behaved unexpectedly."; + case BLE_HS_EBADDATA: return "Command from peer is invalid."; + case BLE_HS_EOS: return "Mynewt OS error."; + case BLE_HS_ECONTROLLER: return "Event from controller is invalid."; + case BLE_HS_ETIMEOUT: return "Operation timed out."; + case BLE_HS_EDONE: return "Operation completed successfully."; + case BLE_HS_EBUSY: return "Operation cannot be performed until procedure completes."; + case BLE_HS_EREJECT: return "Peer rejected a connection parameter update request."; + case BLE_HS_EUNKNOWN: return "Unexpected failure; catch all."; + case BLE_HS_EROLE: return "Operation requires different role (e.g., central vs. peripheral)."; + case BLE_HS_ETIMEOUT_HCI: return "HCI request timed out; controller unresponsive."; + case BLE_HS_ENOMEM_EVT: return "Controller failed to send event due to memory exhaustion (combined host-controller only)."; + case BLE_HS_ENOADDR: return "Operation requires an identity address but none configured."; + case BLE_HS_ENOTSYNCED: return "Attempt to use the host before it is synced with controller."; + case BLE_HS_EAUTHEN: return "Insufficient authentication."; + case BLE_HS_EAUTHOR: return "Insufficient authorization."; + case BLE_HS_EENCRYPT: return "Insufficient encryption level."; + case BLE_HS_EENCRYPT_KEY_SZ: return "Insufficient key size."; + case BLE_HS_ESTORE_CAP: return "Storage at capacity."; + case BLE_HS_ESTORE_FAIL: return "Storage IO error."; + case BLE_HS_EPREEMPTED: return "Operation was preempted."; + case BLE_HS_EDISABLED: return "Operation disabled."; + case BLE_HS_ESTALLED: return "Operation stalled."; + case (0x0100 + BLE_ATT_ERR_INVALID_HANDLE): return "The attribute handle given was not valid on this server."; + case (0x0100 + BLE_ATT_ERR_READ_NOT_PERMITTED): return "The attribute cannot be read."; + case (0x0100 + BLE_ATT_ERR_WRITE_NOT_PERMITTED): return "The attribute cannot be written."; + case (0x0100 + BLE_ATT_ERR_INVALID_PDU): return "The attribute PDU was invalid."; + case (0x0100 + BLE_ATT_ERR_INSUFFICIENT_AUTHEN): return "The attribute requires authentication before it can be read or written."; + case (0x0100 + BLE_ATT_ERR_REQ_NOT_SUPPORTED): return "Attribute server does not support the request received from the client."; + case (0x0100 + BLE_ATT_ERR_INVALID_OFFSET): return "Offset specified was past the end of the attribute."; + case (0x0100 + BLE_ATT_ERR_INSUFFICIENT_AUTHOR): return "The attribute requires authorization before it can be read or written."; + case (0x0100 + BLE_ATT_ERR_PREPARE_QUEUE_FULL): return "Too many prepare writes have been queued."; + case (0x0100 + BLE_ATT_ERR_ATTR_NOT_FOUND): return "No attribute found within the given attribute handle range."; + case (0x0100 + BLE_ATT_ERR_ATTR_NOT_LONG): return "The attribute cannot be read or written using the Read Blob Request."; + case (0x0100 + BLE_ATT_ERR_INSUFFICIENT_KEY_SZ): return "The Encryption Key Size used for encrypting this link is insufficient."; + case (0x0100 + BLE_ATT_ERR_INVALID_ATTR_VALUE_LEN): return "The attribute value length is invalid for the operation."; + case (0x0100 + BLE_ATT_ERR_UNLIKELY): return "The attribute request has encountered an error that was unlikely, could not be completed as requested."; + case (0x0100 + BLE_ATT_ERR_INSUFFICIENT_ENC): return "The attribute requires encryption before it can be read or written."; + case (0x0100 + BLE_ATT_ERR_UNSUPPORTED_GROUP): + return "The attribute type is not a supported grouping attribute as defined by a higher layer specification."; + case (0x0100 + BLE_ATT_ERR_INSUFFICIENT_RES): return "Insufficient Resources to complete the request."; + case (0x0200 + BLE_ERR_UNKNOWN_HCI_CMD): return "Unknown HCI Command"; + case (0x0200 + BLE_ERR_UNK_CONN_ID): return "Unknown Connection Identifier"; + case (0x0200 + BLE_ERR_HW_FAIL): return "Hardware Failure"; + case (0x0200 + BLE_ERR_PAGE_TMO): return "Page Timeout"; + case (0x0200 + BLE_ERR_AUTH_FAIL): return "Authentication Failure"; + case (0x0200 + BLE_ERR_PINKEY_MISSING): return "PIN or Key Missing"; + case (0x0200 + BLE_ERR_MEM_CAPACITY): return "Memory Capacity Exceeded"; + case (0x0200 + BLE_ERR_CONN_SPVN_TMO): return "Connection Timeout"; + case (0x0200 + BLE_ERR_CONN_LIMIT): return "Connection Limit Exceeded"; + case (0x0200 + BLE_ERR_SYNCH_CONN_LIMIT): return "Synchronous Connection Limit To A Device Exceeded"; + case (0x0200 + BLE_ERR_ACL_CONN_EXISTS): return "ACL Connection Already Exists"; + case (0x0200 + BLE_ERR_CMD_DISALLOWED): return "Command Disallowed"; + case (0x0200 + BLE_ERR_CONN_REJ_RESOURCES): return "Connection Rejected due to Limited Resources"; + case (0x0200 + BLE_ERR_CONN_REJ_SECURITY): return "Connection Rejected Due To Security Reasons"; + case (0x0200 + BLE_ERR_CONN_REJ_BD_ADDR): return "Connection Rejected due to Unacceptable BD_ADDR"; + case (0x0200 + BLE_ERR_CONN_ACCEPT_TMO): return "Connection Accept Timeout Exceeded"; + case (0x0200 + BLE_ERR_UNSUPPORTED): return "Unsupported Feature or Parameter Value"; + case (0x0200 + BLE_ERR_INV_HCI_CMD_PARMS): return "Invalid HCI Command Parameters"; + case (0x0200 + BLE_ERR_REM_USER_CONN_TERM): return "Remote User Terminated Connection"; + case (0x0200 + BLE_ERR_RD_CONN_TERM_RESRCS): return "Remote Device Terminated Connection due to Low Resources"; + case (0x0200 + BLE_ERR_RD_CONN_TERM_PWROFF): return "Remote Device Terminated Connection due to Power Off"; + case (0x0200 + BLE_ERR_CONN_TERM_LOCAL): return "Connection Terminated By Local Host"; + case (0x0200 + BLE_ERR_REPEATED_ATTEMPTS): return "Repeated Attempts"; + case (0x0200 + BLE_ERR_NO_PAIRING): return "Pairing Not Allowed"; + case (0x0200 + BLE_ERR_UNK_LMP): return "Unknown LMP PDU"; + case (0x0200 + BLE_ERR_UNSUPP_REM_FEATURE): return "Unsupported Remote Feature / Unsupported LMP Feature"; + case (0x0200 + BLE_ERR_SCO_OFFSET): return "SCO Offset Rejected"; + case (0x0200 + BLE_ERR_SCO_ITVL): return "SCO Interval Rejected"; + case (0x0200 + BLE_ERR_SCO_AIR_MODE): return "SCO Air Mode Rejected"; + case (0x0200 + BLE_ERR_INV_LMP_LL_PARM): return "Invalid LMP Parameters / Invalid LL Parameters"; + case (0x0200 + BLE_ERR_UNSPECIFIED): return "Unspecified Error"; + case (0x0200 + BLE_ERR_UNSUPP_LMP_LL_PARM): return "Unsupported LMP Parameter Value / Unsupported LL Parameter Value"; + case (0x0200 + BLE_ERR_NO_ROLE_CHANGE): return "Role Change Not Allowed"; + case (0x0200 + BLE_ERR_LMP_LL_RSP_TMO): return "LMP Response Timeout / LL Response Timeout"; + case (0x0200 + BLE_ERR_LMP_COLLISION): return "LMP Error Transaction Collision"; + case (0x0200 + BLE_ERR_LMP_PDU): return "LMP PDU Not Allowed"; + case (0x0200 + BLE_ERR_ENCRYPTION_MODE): return "Encryption Mode Not Acceptable"; + case (0x0200 + BLE_ERR_LINK_KEY_CHANGE): return "Link Key cannot be Changed"; + case (0x0200 + BLE_ERR_UNSUPP_QOS): return "Requested QoS Not Supported"; + case (0x0200 + BLE_ERR_INSTANT_PASSED): return "Instant Passed"; + case (0x0200 + BLE_ERR_UNIT_KEY_PAIRING): return "Pairing With Unit Key Not Supported"; + case (0x0200 + BLE_ERR_DIFF_TRANS_COLL): return "Different Transaction Collision"; + case (0x0200 + BLE_ERR_QOS_PARM): return "QoS Unacceptable Parameter"; + case (0x0200 + BLE_ERR_QOS_REJECTED): return "QoS Rejected"; + case (0x0200 + BLE_ERR_CHAN_CLASS): return "Channel Classification Not Supported"; + case (0x0200 + BLE_ERR_INSUFFICIENT_SEC): return "Insufficient Security"; + case (0x0200 + BLE_ERR_PARM_OUT_OF_RANGE): return "Parameter Out Of Mandatory Range"; + case (0x0200 + BLE_ERR_PENDING_ROLE_SW): return "Role Switch Pending"; + case (0x0200 + BLE_ERR_RESERVED_SLOT): return "Reserved Slot Violation"; + case (0x0200 + BLE_ERR_ROLE_SW_FAIL): return "Role Switch Failed"; + case (0x0200 + BLE_ERR_INQ_RSP_TOO_BIG): return "Extended Inquiry Response Too Large"; + case (0x0200 + BLE_ERR_SEC_SIMPLE_PAIR): return "Secure Simple Pairing Not Supported By Host"; + case (0x0200 + BLE_ERR_HOST_BUSY_PAIR): return "Host Busy - Pairing"; + case (0x0200 + BLE_ERR_CONN_REJ_CHANNEL): return "Connection Rejected, No Suitable Channel Found"; + case (0x0200 + BLE_ERR_CTLR_BUSY): return "Controller Busy"; + case (0x0200 + BLE_ERR_CONN_PARMS): return "Unacceptable Connection Parameters"; + case (0x0200 + BLE_ERR_DIR_ADV_TMO): return "Directed Advertising Timeout"; + case (0x0200 + BLE_ERR_CONN_TERM_MIC): return "Connection Terminated due to MIC Failure"; + case (0x0200 + BLE_ERR_CONN_ESTABLISHMENT): return "Connection Failed to be Established"; + case (0x0200 + BLE_ERR_MAC_CONN_FAIL): return "MAC Connection Failed"; + case (0x0200 + BLE_ERR_COARSE_CLK_ADJ): return "Coarse Clock Adjustment Rejected"; + case (0x0300 + BLE_L2CAP_SIG_ERR_CMD_NOT_UNDERSTOOD): return "Invalid or unsupported incoming L2CAP sig command."; + case (0x0300 + BLE_L2CAP_SIG_ERR_MTU_EXCEEDED): return "Incoming packet too large."; + case (0x0300 + BLE_L2CAP_SIG_ERR_INVALID_CID): return "No channel with specified ID."; + case (0x0400 + BLE_SM_ERR_PASSKEY): return "The user input of passkey failed, for example, the user canceled the operation."; + case (0x0400 + BLE_SM_ERR_OOB): return "The OOB data is not available."; + case (0x0400 + BLE_SM_ERR_AUTHREQ): + return "The pairing procedure cannot be performed as authentication requirements cannot be met due to IO capabilities of one or both devices."; + case (0x0400 + BLE_SM_ERR_CONFIRM_MISMATCH): return "The confirm value does not match the calculated compare value."; + case (0x0400 + BLE_SM_ERR_PAIR_NOT_SUPP): return "Pairing is not supported by the device."; + case (0x0400 + BLE_SM_ERR_ENC_KEY_SZ): return "The resultant encryption key size is insufficient for the security requirements of this device."; + case (0x0400 + BLE_SM_ERR_CMD_NOT_SUPP): return "The SMP command received is not supported on this device."; + case (0x0400 + BLE_SM_ERR_UNSPECIFIED): return "Pairing failed due to an unspecified reason."; + case (0x0400 + BLE_SM_ERR_REPEATED): + return "Pairing or authentication procedure disallowed, too little time has elapsed since last pairing request or security request."; + case (0x0400 + BLE_SM_ERR_INVAL): return "Command length is invalid or that a parameter is outside of the specified range."; + case (0x0400 + BLE_SM_ERR_DHKEY): return "DHKey Check value received doesn't match the one calculated by the local device."; + case (0x0400 + BLE_SM_ERR_NUMCMP): return "Confirm values in the numeric comparison protocol do not match."; + case (0x0400 + BLE_SM_ERR_ALREADY): return "Pairing over the LE transport failed - Pairing Request sent over the BR/EDR transport in process."; + case (0x0400 + BLE_SM_ERR_CROSS_TRANS): + return "BR/EDR Link Key generated on the BR/EDR transport cannot be used to derive and distribute keys for the LE transport."; + case (0x0500 + BLE_SM_ERR_PASSKEY): return "The user input of passkey failed or the user canceled the operation."; + case (0x0500 + BLE_SM_ERR_OOB): return "The OOB data is not available."; + case (0x0500 + BLE_SM_ERR_AUTHREQ): + return "The pairing procedure cannot be performed as authentication requirements cannot be met due to IO capabilities of one or both devices."; + case (0x0500 + BLE_SM_ERR_CONFIRM_MISMATCH): return "The confirm value does not match the calculated compare value."; + case (0x0500 + BLE_SM_ERR_PAIR_NOT_SUPP): return "Pairing is not supported by the device."; + case (0x0500 + BLE_SM_ERR_ENC_KEY_SZ): return "The resultant encryption key size is insufficient for the security requirements of this device."; + case (0x0500 + BLE_SM_ERR_CMD_NOT_SUPP): return "The SMP command received is not supported on this device."; + case (0x0500 + BLE_SM_ERR_UNSPECIFIED): return "Pairing failed due to an unspecified reason."; + case (0x0500 + BLE_SM_ERR_REPEATED): + return "Pairing or authentication procedure is disallowed because too little time has elapsed since last pairing request or security request."; + case (0x0500 + BLE_SM_ERR_INVAL): return "Command length is invalid or a parameter is outside of the specified range."; + case (0x0500 + BLE_SM_ERR_DHKEY): + return "Indicates to the remote device that the DHKey Check value received doesn't match the one calculated by the local device."; + case (0x0500 + BLE_SM_ERR_NUMCMP): return "Confirm values in the numeric comparison protocol do not match."; + case (0x0500 + BLE_SM_ERR_ALREADY): return "Pairing over the LE transport failed - Pairing Request sent over the BR/EDR transport in process."; + case (0x0500 + BLE_SM_ERR_CROSS_TRANS): + return "BR/EDR Link Key generated on the BR/EDR transport cannot be used to derive and distribute keys for the LE transport."; + default: return "Unknown"; + } +#else // #if defined(CONFIG_NIMBLE_ENABLE_RETURN_CODE_TEXT) + return ""; +#endif // #if defined(CONFIG_NIMBLE_ENABLE_RETURN_CODE_TEXT) +} + +/** + * @brief Utility function to log the gap event info. + * @param [in] event A pointer to the gap event structure. + * @param [in] arg Unused. + */ +void BLEUtils::dumpGapEvent(ble_gap_event *event, void *arg) { +#if defined(CONFIG_NIMBLE_ENABLE_GAP_EVENT_CODE_TEXT) + log_d("Received a GAP event: %s", gapEventToString(event->type)); +#endif +} + +/** + * @brief Convert a GAP event type to a string representation. + * @param [in] eventType The type of event. + * @return A string representation of the event type. + */ +const char *BLEUtils::gapEventToString(uint8_t eventType) { +#if defined(CONFIG_NIMBLE_ENABLE_GAP_EVENT_CODE_TEXT) + switch (eventType) { + case BLE_GAP_EVENT_CONNECT: //0 + return "BLE_GAP_EVENT_CONNECT "; + + case BLE_GAP_EVENT_DISCONNECT: //1 + return "BLE_GAP_EVENT_DISCONNECT"; + + case BLE_GAP_EVENT_CONN_UPDATE: //3 + return "BLE_GAP_EVENT_CONN_UPDATE"; + + case BLE_GAP_EVENT_CONN_UPDATE_REQ: //4 + return "BLE_GAP_EVENT_CONN_UPDATE_REQ"; + + case BLE_GAP_EVENT_L2CAP_UPDATE_REQ: //5 + return "BLE_GAP_EVENT_L2CAP_UPDATE_REQ"; + + case BLE_GAP_EVENT_TERM_FAILURE: //6 + return "BLE_GAP_EVENT_TERM_FAILURE"; + + case BLE_GAP_EVENT_DISC: //7 + return "BLE_GAP_EVENT_DISC"; + + case BLE_GAP_EVENT_DISC_COMPLETE: //8 + return "BLE_GAP_EVENT_DISC_COMPLETE"; + + case BLE_GAP_EVENT_ADV_COMPLETE: //9 + return "BLE_GAP_EVENT_ADV_COMPLETE"; + + case BLE_GAP_EVENT_ENC_CHANGE: //10 + return "BLE_GAP_EVENT_ENC_CHANGE"; + + case BLE_GAP_EVENT_PASSKEY_ACTION: //11 + return "BLE_GAP_EVENT_PASSKEY_ACTION"; + + case BLE_GAP_EVENT_NOTIFY_RX: //12 + return "BLE_GAP_EVENT_NOTIFY_RX"; + + case BLE_GAP_EVENT_NOTIFY_TX: //13 + return "BLE_GAP_EVENT_NOTIFY_TX"; + + case BLE_GAP_EVENT_SUBSCRIBE: //14 + return "BLE_GAP_EVENT_SUBSCRIBE"; + + case BLE_GAP_EVENT_MTU: //15 + return "BLE_GAP_EVENT_MTU"; + + case BLE_GAP_EVENT_IDENTITY_RESOLVED: //16 + return "BLE_GAP_EVENT_IDENTITY_RESOLVED"; + + case BLE_GAP_EVENT_REPEAT_PAIRING: //17 + return "BLE_GAP_EVENT_REPEAT_PAIRING"; + + case BLE_GAP_EVENT_PHY_UPDATE_COMPLETE: //18 + return "BLE_GAP_EVENT_PHY_UPDATE_COMPLETE"; + + case BLE_GAP_EVENT_EXT_DISC: //19 + return "BLE_GAP_EVENT_EXT_DISC"; +#ifdef BLE_GAP_EVENT_PERIODIC_SYNC // IDF 4.0 does not support these + case BLE_GAP_EVENT_PERIODIC_SYNC: //20 + return "BLE_GAP_EVENT_PERIODIC_SYNC"; + + case BLE_GAP_EVENT_PERIODIC_REPORT: //21 + return "BLE_GAP_EVENT_PERIODIC_REPORT"; + + case BLE_GAP_EVENT_PERIODIC_SYNC_LOST: //22 + return "BLE_GAP_EVENT_PERIODIC_SYNC_LOST"; + + case BLE_GAP_EVENT_SCAN_REQ_RCVD: //23 + return "BLE_GAP_EVENT_SCAN_REQ_RCVD"; +#endif + default: log_d("gapEventToString: Unknown event type %d 0x%.2x", eventType, eventType); return "Unknown event type"; + } +#else // #if defined(CONFIG_NIMBLE_ENABLE_GAP_EVENT_CODE_TEXT) + return ""; +#endif // #if defined(CONFIG_NIMBLE_ENABLE_GAP_EVENT_CODE_TEXT) +} // gapEventToString + +/** + * @brief Convert characteristic properties into a string representation. + * @param [in] prop Characteristic properties. + * @return A string representation of characteristic properties. + */ +String BLEUtils::characteristicPropertiesToString(uint8_t prop) { + String res = "broadcast: "; + res += ((prop & BLE_GATT_CHR_PROP_BROADCAST) ? "1" : "0"); + res += ", read: "; + res += ((prop & BLE_GATT_CHR_PROP_READ) ? "1" : "0"); + res += ", write_nr: "; + res += ((prop & BLE_GATT_CHR_PROP_WRITE_NO_RSP) ? "1" : "0"); + res += ", write: "; + res += ((prop & BLE_GATT_CHR_PROP_WRITE) ? "1" : "0"); + res += ", notify: "; + res += ((prop & BLE_GATT_CHR_PROP_NOTIFY) ? "1" : "0"); + res += ", indicate: "; + res += ((prop & BLE_GATT_CHR_PROP_INDICATE) ? "1" : "0"); + res += ", auth_sign_write: "; + res += ((prop & BLE_GATT_CHR_PROP_AUTH_SIGN_WRITE) ? "1" : "0"); + res += ", extended: "; + res += ((prop & BLE_GATT_CHR_PROP_EXTENDED) ? "1" : "0"); + return res; +} // characteristicPropertiesToString + +/** + * @brief Blocks the calling task until released or timeout. + * @param [in] taskData A pointer to the task data structure. + * @param [in] timeout The time to wait in milliseconds. + * @return True if the task completed, false if the timeout was reached. + */ +bool BLEUtils::taskWait(const BLETaskData &taskData, uint32_t timeout) { + ble_npl_time_t ticks; + if (timeout == BLE_NPL_TIME_FOREVER) { + ticks = BLE_NPL_TIME_FOREVER; + } else { + ble_npl_time_ms_to_ticks(timeout, &ticks); + } + + uint32_t notificationValue; + xTaskNotifyWait(0, TASK_BLOCK_BIT, ¬ificationValue, 0); + if (notificationValue & TASK_BLOCK_BIT) { + return true; + } + + return xTaskNotifyWait(0, TASK_BLOCK_BIT, nullptr, ticks) == pdTRUE; +} // taskWait + +/** + * @brief Release a task. + * @param [in] taskData A pointer to the task data structure. + * @param [in] flags A return value to set in the task data structure. + */ +void BLEUtils::taskRelease(const BLETaskData &taskData, int flags) { + taskData.m_flags = flags; + if (taskData.m_pHandle != nullptr) { + xTaskNotify(static_cast(taskData.m_pHandle), TASK_BLOCK_BIT, eSetBits); + } +} // taskRelease + +#endif // CONFIG_NIMBLE_ENABLED + +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ diff --git a/libraries/BLE/src/BLEUtils.h b/libraries/BLE/src/BLEUtils.h index 7c6f58d284b..2689706b7a1 100644 --- a/libraries/BLE/src/BLEUtils.h +++ b/libraries/BLE/src/BLEUtils.h @@ -3,6 +3,10 @@ * * Created on: Mar 25, 2017 * Author: kolban + * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on kolban's and h2zero's work) + * Description: Added support for NimBLE */ #ifndef COMPONENTS_CPP_UTILS_BLEUTILS_H_ @@ -11,24 +15,99 @@ #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) -#include // ESP32 BLE -#include // ESP32 BLE -#include // ESP32 BLE +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) + +/***************************************************************************** + * Common includes * + *****************************************************************************/ + #include -#include "BLEClient.h" +#include "BLEAddress.h" +#include "WString.h" +#include + +/***************************************************************************** + * Bluedroid includes * + *****************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) +#include +#include +#include +#endif + +/***************************************************************************** + * NimBLE includes * + *****************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) +#include +#include +#include +#endif + +/***************************************************************************** + * Common types * + *****************************************************************************/ + +typedef struct { + void *peer_device; // peer device BLEClient or BLEServer + bool connected; // connection status + uint16_t mtu; // negotiated MTU per peer device +} conn_status_t; + +/***************************************************************************** + * NimBLE types * + *****************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) + +/** + * @brief A structure to hold data for a task that is waiting for a response. + * @details This structure is used in conjunction with BLEUtils::taskWait() and BLEUtils::taskRelease(). + * All items are optional, the m_pHandle will be set in taskWait(). + */ +struct BLETaskData { + BLETaskData(void *pInstance = nullptr, int flags = 0, void *buf = nullptr); + ~BLETaskData(); + void *m_pInstance{nullptr}; + mutable int m_flags{0}; + void *m_pBuf{nullptr}; + +private: + mutable void *m_pHandle{nullptr}; // semaphore or task handle + friend class BLEUtils; +}; + +#endif + +/***************************************************************************** + * Forward declarations * + *****************************************************************************/ + +class BLEClient; /** * @brief A set of general %BLE utilities. */ class BLEUtils { public: - static const char *addressTypeToString(esp_ble_addr_type_t type); - static String adFlagsToString(uint8_t adFlags); - static const char *advTypeToString(uint8_t advType); + /*************************************************************************** + * Common public declarations * + ***************************************************************************/ + static char *buildHexData(uint8_t *target, uint8_t *source, uint8_t length); static String buildPrintData(uint8_t *source, size_t length); - static String characteristicPropertiesToString(esp_gatt_char_prop_t prop); + static const char *advDataTypeToString(uint8_t advType); + static String characteristicPropertiesToString(uint8_t prop); + + /*************************************************************************** + * Bluedroid public declarations * + ***************************************************************************/ + +#if defined(CONFIG_BLUEDROID_ENABLED) + static const char *addressTypeToString(esp_ble_addr_type_t type); + static String adFlagsToString(uint8_t adFlags); static const char *devTypeToString(esp_bt_dev_type_t type); static esp_gatt_id_t buildGattId(esp_bt_uuid_t uuid, uint8_t inst_id = 0); static esp_gatt_srvc_id_t buildGattSrvcId(esp_gatt_id_t gattId, bool is_primary = true); @@ -52,8 +131,22 @@ class BLEUtils { static void registerByAddress(BLEAddress address, BLEClient *pDevice); static void registerByConnId(uint16_t conn_id, BLEClient *pDevice); static const char *searchEventTypeToString(esp_gap_search_evt_t searchEvt); +#endif + + /*************************************************************************** + * NimBLE public declarations * + ***************************************************************************/ + +#if defined(CONFIG_NIMBLE_ENABLED) + static void dumpGapEvent(ble_gap_event *event, void *arg); + static const char *gapEventToString(uint8_t eventType); + static const char *returnCodeToString(int rc); + static int checkConnParams(ble_gap_conn_params *params); + static bool taskWait(const BLETaskData &taskData, uint32_t timeout); + static void taskRelease(const BLETaskData &taskData, int rc = 0); +#endif }; -#endif /* CONFIG_BLUEDROID_ENABLED */ +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ #endif /* COMPONENTS_CPP_UTILS_BLEUTILS_H_ */ diff --git a/libraries/BLE/src/BLEValue.cpp b/libraries/BLE/src/BLEValue.cpp index 26811c985ac..efc97697baa 100644 --- a/libraries/BLE/src/BLEValue.cpp +++ b/libraries/BLE/src/BLEValue.cpp @@ -3,15 +3,29 @@ * * Created on: Jul 17, 2017 * Author: kolban + * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on kolban's and h2zero's work) + * Description: Added support for NimBLE */ + #include "soc/soc_caps.h" #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) + +/***************************************************************************** + * Common includes * + *****************************************************************************/ + #include "BLEValue.h" #include "esp32-hal-log.h" +/***************************************************************************** + * Common functions * + *****************************************************************************/ + BLEValue::BLEValue() { m_accumulation = ""; m_value = ""; @@ -120,5 +134,5 @@ void BLEValue::setValue(uint8_t *pData, size_t length) { m_value = String((char *)pData, length); } // setValue -#endif /* CONFIG_BLUEDROID_ENABLED */ +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ diff --git a/libraries/BLE/src/BLEValue.h b/libraries/BLE/src/BLEValue.h index f9c91bdcd4e..56a7a5bc4ec 100644 --- a/libraries/BLE/src/BLEValue.h +++ b/libraries/BLE/src/BLEValue.h @@ -3,22 +3,36 @@ * * Created on: Jul 17, 2017 * Author: kolban + * + * Modified on: Feb 18, 2025 + * Author: lucasssvaz (based on kolban's and h2zero's work) + * Description: Added support for NimBLE */ #ifndef COMPONENTS_CPP_UTILS_BLEVALUE_H_ #define COMPONENTS_CPP_UTILS_BLEVALUE_H_ + #include "soc/soc_caps.h" -#include "WString.h" #if SOC_BLE_SUPPORTED #include "sdkconfig.h" -#if defined(CONFIG_BLUEDROID_ENABLED) +#if defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED) + +/***************************************************************************** + * Common includes * + *****************************************************************************/ + +#include "WString.h" /** * @brief The model of a %BLE value. */ class BLEValue { public: + /*************************************************************************** + * Common public declarations * + ***************************************************************************/ + BLEValue(); void addPart(String part); void addPart(uint8_t *pData, size_t length); @@ -33,10 +47,15 @@ class BLEValue { void setValue(uint8_t *pData, size_t length); private: + /*************************************************************************** + * Common private properties * + ***************************************************************************/ + String m_accumulation; uint16_t m_readOffset; String m_value; }; -#endif /* CONFIG_BLUEDROID_ENABLED */ + +#endif /* CONFIG_BLUEDROID_ENABLED || CONFIG_NIMBLE_ENABLED */ #endif /* SOC_BLE_SUPPORTED */ #endif /* COMPONENTS_CPP_UTILS_BLEVALUE_H_ */ From dc82467ba41c7ed589dfb480db742ea14e65b68f Mon Sep 17 00:00:00 2001 From: Lucas Saavedra Vaz <32426024+lucasssvaz@users.noreply.github.com> Date: Mon, 23 Jun 2025 08:48:37 -0300 Subject: [PATCH 064/173] fix(esp_now): Fix broadcast example and use nullptr (#11490) * fix(esp_now): Fix example and use nullptr * ci(pre-commit): Apply automatic fixes --------- Co-authored-by: pre-commit-ci-lite[bot] <117423508+pre-commit-ci-lite[bot]@users.noreply.github.com> --- .../ESP_NOW_Broadcast_Master.ino | 2 +- .../ESP_NOW_Broadcast_Slave.ino | 28 ++++++++--- .../ESP_NOW_Network/ESP_NOW_Network.ino | 13 +++-- libraries/ESP_NOW/src/ESP32_NOW.cpp | 42 ++++++++-------- libraries/ESP_NOW/src/ESP32_NOW.h | 4 +- libraries/ESP_NOW/src/ESP32_NOW_Serial.cpp | 50 +++++++++---------- libraries/ESP_NOW/src/ESP32_NOW_Serial.h | 4 +- 7 files changed, 81 insertions(+), 62 deletions(-) diff --git a/libraries/ESP_NOW/examples/ESP_NOW_Broadcast_Master/ESP_NOW_Broadcast_Master.ino b/libraries/ESP_NOW/examples/ESP_NOW_Broadcast_Master/ESP_NOW_Broadcast_Master.ino index 6ee7dc6f846..025a53c913b 100644 --- a/libraries/ESP_NOW/examples/ESP_NOW_Broadcast_Master/ESP_NOW_Broadcast_Master.ino +++ b/libraries/ESP_NOW/examples/ESP_NOW_Broadcast_Master/ESP_NOW_Broadcast_Master.ino @@ -58,7 +58,7 @@ public: uint32_t msg_count = 0; // Create a broadcast peer object -ESP_NOW_Broadcast_Peer broadcast_peer(ESPNOW_WIFI_CHANNEL, WIFI_IF_STA, NULL); +ESP_NOW_Broadcast_Peer broadcast_peer(ESPNOW_WIFI_CHANNEL, WIFI_IF_STA, nullptr); /* Main */ diff --git a/libraries/ESP_NOW/examples/ESP_NOW_Broadcast_Slave/ESP_NOW_Broadcast_Slave.ino b/libraries/ESP_NOW/examples/ESP_NOW_Broadcast_Slave/ESP_NOW_Broadcast_Slave.ino index 5f1a8bd8807..e61524b64f9 100644 --- a/libraries/ESP_NOW/examples/ESP_NOW_Broadcast_Slave/ESP_NOW_Broadcast_Slave.ino +++ b/libraries/ESP_NOW/examples/ESP_NOW_Broadcast_Slave/ESP_NOW_Broadcast_Slave.ino @@ -52,7 +52,8 @@ public: /* Global Variables */ // List of all the masters. It will be populated when a new master is registered -std::vector masters; +// Note: Using pointers instead of objects to prevent dangling pointers when the vector reallocates +std::vector masters; /* Callbacks */ @@ -62,13 +63,14 @@ void register_new_master(const esp_now_recv_info_t *info, const uint8_t *data, i Serial.printf("Unknown peer " MACSTR " sent a broadcast message\n", MAC2STR(info->src_addr)); Serial.println("Registering the peer as a master"); - ESP_NOW_Peer_Class new_master(info->src_addr, ESPNOW_WIFI_CHANNEL, WIFI_IF_STA, NULL); - - masters.push_back(new_master); - if (!masters.back().add_peer()) { + ESP_NOW_Peer_Class *new_master = new ESP_NOW_Peer_Class(info->src_addr, ESPNOW_WIFI_CHANNEL, WIFI_IF_STA, nullptr); + if (!new_master->add_peer()) { Serial.println("Failed to register the new master"); + delete new_master; return; } + masters.push_back(new_master); + Serial.printf("Successfully registered master " MACSTR " (total masters: %zu)\n", MAC2STR(new_master->addr()), masters.size()); } else { // The slave will only receive broadcast messages log_v("Received a unicast message from " MACSTR, MAC2STR(info->src_addr)); @@ -103,11 +105,23 @@ void setup() { } // Register the new peer callback - ESP_NOW.onNewPeer(register_new_master, NULL); + ESP_NOW.onNewPeer(register_new_master, nullptr); Serial.println("Setup complete. Waiting for a master to broadcast a message..."); } void loop() { - delay(1000); + // Print debug information every 10 seconds + static unsigned long last_debug = 0; + if (millis() - last_debug > 10000) { + last_debug = millis(); + Serial.printf("Registered masters: %zu\n", masters.size()); + for (size_t i = 0; i < masters.size(); i++) { + if (masters[i]) { + Serial.printf(" Master %zu: " MACSTR "\n", i, MAC2STR(masters[i]->addr())); + } + } + } + + delay(100); } diff --git a/libraries/ESP_NOW/examples/ESP_NOW_Network/ESP_NOW_Network.ino b/libraries/ESP_NOW/examples/ESP_NOW_Network/ESP_NOW_Network.ino index 48ea6b731ce..6731340c922 100644 --- a/libraries/ESP_NOW/examples/ESP_NOW_Network/ESP_NOW_Network.ino +++ b/libraries/ESP_NOW/examples/ESP_NOW_Network/ESP_NOW_Network.ino @@ -123,7 +123,7 @@ public: } bool send_message(const uint8_t *data, size_t len) { - if (data == NULL || len == 0) { + if (data == nullptr || len == 0) { log_e("Data to be sent is NULL or has a length of 0"); return false; } @@ -169,9 +169,12 @@ public: /* Peers */ -std::vector peers; // Create a vector to store the peer pointers -ESP_NOW_Network_Peer broadcast_peer(ESP_NOW.BROADCAST_ADDR, 0, NULL); // Register the broadcast peer (no encryption support for the broadcast address) -ESP_NOW_Network_Peer *master_peer = nullptr; // Pointer to peer that is the master +// Create a vector to store the peer pointers +std::vector peers; +// Register the broadcast peer (no encryption support for the broadcast address) +ESP_NOW_Network_Peer broadcast_peer(ESP_NOW.BROADCAST_ADDR, 0, nullptr); +// Pointer to the peer that is the master +ESP_NOW_Network_Peer *master_peer = nullptr; /* Helper functions */ @@ -279,7 +282,7 @@ void setup() { } // Register the callback to be called when a new peer is found - ESP_NOW.onNewPeer(register_new_peer, NULL); + ESP_NOW.onNewPeer(register_new_peer, nullptr); Serial.println("Setup complete. Broadcasting own priority to find the master..."); memset(&new_msg, 0, sizeof(new_msg)); diff --git a/libraries/ESP_NOW/src/ESP32_NOW.cpp b/libraries/ESP_NOW/src/ESP32_NOW.cpp index 6fd3ff0a0b1..25a5609e8db 100644 --- a/libraries/ESP_NOW/src/ESP32_NOW.cpp +++ b/libraries/ESP_NOW/src/ESP32_NOW.cpp @@ -9,12 +9,12 @@ #include "esp32-hal.h" #include "esp_wifi.h" -static void (*new_cb)(const esp_now_recv_info_t *info, const uint8_t *data, int len, void *arg) = NULL; -static void *new_arg = NULL; // * tx_arg = NULL, * rx_arg = NULL, +static void (*new_cb)(const esp_now_recv_info_t *info, const uint8_t *data, int len, void *arg) = nullptr; +static void *new_arg = nullptr; // * tx_arg = nullptr, * rx_arg = nullptr, static bool _esp_now_has_begun = false; static ESP_NOW_Peer *_esp_now_peers[ESP_NOW_MAX_TOTAL_PEER_NUM]; -static esp_err_t _esp_now_add_peer(const uint8_t *mac_addr, uint8_t channel, wifi_interface_t iface, const uint8_t *lmk, ESP_NOW_Peer *_peer = NULL) { +static esp_err_t _esp_now_add_peer(const uint8_t *mac_addr, uint8_t channel, wifi_interface_t iface, const uint8_t *lmk, ESP_NOW_Peer *_peer = nullptr) { log_v(MACSTR, MAC2STR(mac_addr)); if (esp_now_is_peer_exist(mac_addr)) { log_e("Peer Already Exists"); @@ -26,16 +26,16 @@ static esp_err_t _esp_now_add_peer(const uint8_t *mac_addr, uint8_t channel, wif memcpy(peer.peer_addr, mac_addr, ESP_NOW_ETH_ALEN); peer.channel = channel; peer.ifidx = iface; - peer.encrypt = lmk != NULL; + peer.encrypt = lmk != nullptr; if (lmk) { memcpy(peer.lmk, lmk, ESP_NOW_KEY_LEN); } esp_err_t result = esp_now_add_peer(&peer); if (result == ESP_OK) { - if (_peer != NULL) { + if (_peer != nullptr) { for (uint8_t i = 0; i < ESP_NOW_MAX_TOTAL_PEER_NUM; i++) { - if (_esp_now_peers[i] == NULL) { + if (_esp_now_peers[i] == nullptr) { _esp_now_peers[i] = _peer; return ESP_OK; } @@ -67,8 +67,8 @@ static esp_err_t _esp_now_del_peer(const uint8_t *mac_addr) { } for (uint8_t i = 0; i < ESP_NOW_MAX_TOTAL_PEER_NUM; i++) { - if (_esp_now_peers[i] != NULL && memcmp(mac_addr, _esp_now_peers[i]->addr(), ESP_NOW_ETH_ALEN) == 0) { - _esp_now_peers[i] = NULL; + if (_esp_now_peers[i] != nullptr && memcmp(mac_addr, _esp_now_peers[i]->addr(), ESP_NOW_ETH_ALEN) == 0) { + _esp_now_peers[i] = nullptr; break; } } @@ -87,7 +87,7 @@ static esp_err_t _esp_now_modify_peer(const uint8_t *mac_addr, uint8_t channel, memcpy(peer.peer_addr, mac_addr, ESP_NOW_ETH_ALEN); peer.channel = channel; peer.ifidx = iface; - peer.encrypt = lmk != NULL; + peer.encrypt = lmk != nullptr; if (lmk) { memcpy(peer.lmk, lmk, ESP_NOW_KEY_LEN); } @@ -111,17 +111,17 @@ static void _esp_now_rx_cb(const esp_now_recv_info_t *info, const uint8_t *data, bool broadcast = memcmp(info->des_addr, ESP_NOW.BROADCAST_ADDR, ESP_NOW_ETH_ALEN) == 0; log_v("%s from " MACSTR ", data length : %u", broadcast ? "Broadcast" : "Unicast", MAC2STR(info->src_addr), len); log_buf_v(data, len); - if (!esp_now_is_peer_exist(info->src_addr) && new_cb != NULL) { + if (!esp_now_is_peer_exist(info->src_addr) && new_cb != nullptr) { log_v("Calling new_cb, peer not found."); new_cb(info, data, len, new_arg); return; } //find the peer and call it's callback for (uint8_t i = 0; i < ESP_NOW_MAX_TOTAL_PEER_NUM; i++) { - if (_esp_now_peers[i] != NULL) { + if (_esp_now_peers[i] != nullptr) { log_v("Checking peer " MACSTR, MAC2STR(_esp_now_peers[i]->addr())); } - if (_esp_now_peers[i] != NULL && memcmp(info->src_addr, _esp_now_peers[i]->addr(), ESP_NOW_ETH_ALEN) == 0) { + if (_esp_now_peers[i] != nullptr && memcmp(info->src_addr, _esp_now_peers[i]->addr(), ESP_NOW_ETH_ALEN) == 0) { log_v("Calling onReceive"); _esp_now_peers[i]->onReceive(data, len, broadcast); return; @@ -133,7 +133,7 @@ static void _esp_now_tx_cb(const uint8_t *mac_addr, esp_now_send_status_t status log_v(MACSTR " : %s", MAC2STR(mac_addr), (status == ESP_NOW_SEND_SUCCESS) ? "SUCCESS" : "FAILED"); //find the peer and call it's callback for (uint8_t i = 0; i < ESP_NOW_MAX_TOTAL_PEER_NUM; i++) { - if (_esp_now_peers[i] != NULL && memcmp(mac_addr, _esp_now_peers[i]->addr(), ESP_NOW_ETH_ALEN) == 0) { + if (_esp_now_peers[i] != nullptr && memcmp(mac_addr, _esp_now_peers[i]->addr(), ESP_NOW_ETH_ALEN) == 0) { _esp_now_peers[i]->onSent(status == ESP_NOW_SEND_SUCCESS); return; } @@ -197,7 +197,7 @@ bool ESP_NOW_Class::end() { } //remove all peers for (uint8_t i = 0; i < ESP_NOW_MAX_TOTAL_PEER_NUM; i++) { - if (_esp_now_peers[i] != NULL) { + if (_esp_now_peers[i] != nullptr) { removePeer(*_esp_now_peers[i]); } } @@ -249,7 +249,7 @@ size_t ESP_NOW_Class::write(const uint8_t *data, size_t len) { if (len > ESP_NOW_MAX_DATA_LEN) { len = ESP_NOW_MAX_DATA_LEN; } - esp_err_t result = esp_now_send(NULL, data, len); + esp_err_t result = esp_now_send(nullptr, data, len); if (result == ESP_OK) { return len; } else if (result == ESP_ERR_ESPNOW_NOT_INIT) { @@ -292,7 +292,7 @@ ESP_NOW_Peer::ESP_NOW_Peer(const uint8_t *mac_addr, uint8_t channel, wifi_interf } chan = channel; ifc = iface; - encrypt = lmk != NULL; + encrypt = lmk != nullptr; if (encrypt) { memcpy(key, lmk, 16); } @@ -305,7 +305,7 @@ bool ESP_NOW_Peer::add() { if (added) { return true; } - if (_esp_now_add_peer(mac, chan, ifc, encrypt ? key : NULL, this) != ESP_OK) { + if (_esp_now_add_peer(mac, chan, ifc, encrypt ? key : nullptr, this) != ESP_OK) { return false; } log_v("Peer added - " MACSTR, MAC2STR(mac)); @@ -350,7 +350,7 @@ bool ESP_NOW_Peer::setChannel(uint8_t channel) { if (!_esp_now_has_begun || !added) { return true; } - return _esp_now_modify_peer(mac, chan, ifc, encrypt ? key : NULL) == ESP_OK; + return _esp_now_modify_peer(mac, chan, ifc, encrypt ? key : nullptr) == ESP_OK; } wifi_interface_t ESP_NOW_Peer::getInterface() const { @@ -362,7 +362,7 @@ bool ESP_NOW_Peer::setInterface(wifi_interface_t iface) { if (!_esp_now_has_begun || !added) { return true; } - return _esp_now_modify_peer(mac, chan, ifc, encrypt ? key : NULL) == ESP_OK; + return _esp_now_modify_peer(mac, chan, ifc, encrypt ? key : nullptr) == ESP_OK; } bool ESP_NOW_Peer::isEncrypted() const { @@ -370,14 +370,14 @@ bool ESP_NOW_Peer::isEncrypted() const { } bool ESP_NOW_Peer::setKey(const uint8_t *lmk) { - encrypt = lmk != NULL; + encrypt = lmk != nullptr; if (encrypt) { memcpy(key, lmk, 16); } if (!_esp_now_has_begun || !added) { return true; } - return _esp_now_modify_peer(mac, chan, ifc, encrypt ? key : NULL) == ESP_OK; + return _esp_now_modify_peer(mac, chan, ifc, encrypt ? key : nullptr) == ESP_OK; } size_t ESP_NOW_Peer::send(const uint8_t *data, int len) { diff --git a/libraries/ESP_NOW/src/ESP32_NOW.h b/libraries/ESP_NOW/src/ESP32_NOW.h index efba9243aee..5940cfa2221 100644 --- a/libraries/ESP_NOW/src/ESP32_NOW.h +++ b/libraries/ESP_NOW/src/ESP32_NOW.h @@ -20,7 +20,7 @@ class ESP_NOW_Class : public Print { ESP_NOW_Class(); ~ESP_NOW_Class(); - bool begin(const uint8_t *pmk = NULL /* 16 bytes */); + bool begin(const uint8_t *pmk = nullptr /* 16 bytes */); bool end(); int getTotalPeerCount(); @@ -50,7 +50,7 @@ class ESP_NOW_Peer { bool remove(); size_t send(const uint8_t *data, int len); - ESP_NOW_Peer(const uint8_t *mac_addr, uint8_t channel = 0, wifi_interface_t iface = WIFI_IF_AP, const uint8_t *lmk = NULL); + ESP_NOW_Peer(const uint8_t *mac_addr, uint8_t channel = 0, wifi_interface_t iface = WIFI_IF_AP, const uint8_t *lmk = nullptr); public: virtual ~ESP_NOW_Peer() {} diff --git a/libraries/ESP_NOW/src/ESP32_NOW_Serial.cpp b/libraries/ESP_NOW/src/ESP32_NOW_Serial.cpp index 5603da2ba13..edd6e32aacc 100644 --- a/libraries/ESP_NOW/src/ESP32_NOW_Serial.cpp +++ b/libraries/ESP_NOW/src/ESP32_NOW_Serial.cpp @@ -18,11 +18,11 @@ ESP_NOW_Serial_Class::ESP_NOW_Serial_Class(const uint8_t *mac_addr, uint8_t channel, wifi_interface_t iface, const uint8_t *lmk, bool remove_on_fail) : ESP_NOW_Peer(mac_addr, channel, iface, lmk) { - tx_ring_buf = NULL; - rx_queue = NULL; - tx_sem = NULL; + tx_ring_buf = nullptr; + rx_queue = nullptr; + tx_sem = nullptr; queued_size = 0; - queued_buff = NULL; + queued_buff = nullptr; resend_count = 0; _remove_on_fail = remove_on_fail; } @@ -34,7 +34,7 @@ ESP_NOW_Serial_Class::~ESP_NOW_Serial_Class() { size_t ESP_NOW_Serial_Class::setTxBufferSize(size_t tx_queue_len) { if (tx_ring_buf) { vRingbufferDelete(tx_ring_buf); - tx_ring_buf = NULL; + tx_ring_buf = nullptr; } if (!tx_queue_len) { return 0; @@ -49,7 +49,7 @@ size_t ESP_NOW_Serial_Class::setTxBufferSize(size_t tx_queue_len) { size_t ESP_NOW_Serial_Class::setRxBufferSize(size_t rx_queue_len) { if (rx_queue) { vQueueDelete(rx_queue); - rx_queue = NULL; + rx_queue = nullptr; } if (!rx_queue_len) { return 0; @@ -65,7 +65,7 @@ bool ESP_NOW_Serial_Class::begin(unsigned long baud) { if (!ESP_NOW.begin() || !add()) { return false; } - if (tx_sem == NULL) { + if (tx_sem == nullptr) { tx_sem = xSemaphoreCreateBinary(); //xSemaphoreTake(tx_sem, 0); xSemaphoreGive(tx_sem); @@ -79,22 +79,22 @@ void ESP_NOW_Serial_Class::end() { remove(); setRxBufferSize(0); setTxBufferSize(0); - if (tx_sem != NULL) { + if (tx_sem != nullptr) { vSemaphoreDelete(tx_sem); - tx_sem = NULL; + tx_sem = nullptr; } } //Stream int ESP_NOW_Serial_Class::available(void) { - if (rx_queue == NULL) { + if (rx_queue == nullptr) { return 0; } return uxQueueMessagesWaiting(rx_queue); } int ESP_NOW_Serial_Class::peek(void) { - if (rx_queue == NULL) { + if (rx_queue == nullptr) { return -1; } uint8_t c; @@ -105,7 +105,7 @@ int ESP_NOW_Serial_Class::peek(void) { } int ESP_NOW_Serial_Class::read(void) { - if (rx_queue == NULL) { + if (rx_queue == nullptr) { return -1; } uint8_t c = 0; @@ -116,7 +116,7 @@ int ESP_NOW_Serial_Class::read(void) { } size_t ESP_NOW_Serial_Class::read(uint8_t *buffer, size_t size) { - if (rx_queue == NULL) { + if (rx_queue == nullptr) { return -1; } uint8_t c = 0; @@ -128,11 +128,11 @@ size_t ESP_NOW_Serial_Class::read(uint8_t *buffer, size_t size) { } void ESP_NOW_Serial_Class::flush() { - if (tx_ring_buf == NULL) { + if (tx_ring_buf == nullptr) { return; } UBaseType_t uxItemsWaiting = 0; - vRingbufferGetInfo(tx_ring_buf, NULL, NULL, NULL, NULL, &uxItemsWaiting); + vRingbufferGetInfo(tx_ring_buf, nullptr, nullptr, nullptr, nullptr, &uxItemsWaiting); if (uxItemsWaiting) { // Now trigger the ISR to read data from the ring buffer. if (xSemaphoreTake(tx_sem, 0) == pdTRUE) { @@ -141,13 +141,13 @@ void ESP_NOW_Serial_Class::flush() { } while (uxItemsWaiting) { delay(5); - vRingbufferGetInfo(tx_ring_buf, NULL, NULL, NULL, NULL, &uxItemsWaiting); + vRingbufferGetInfo(tx_ring_buf, nullptr, nullptr, nullptr, nullptr, &uxItemsWaiting); } } //RX callback void ESP_NOW_Serial_Class::onReceive(const uint8_t *data, size_t len, bool broadcast) { - if (rx_queue == NULL) { + if (rx_queue == nullptr) { return; } for (uint32_t i = 0; i < len; i++) { @@ -165,7 +165,7 @@ void ESP_NOW_Serial_Class::onReceive(const uint8_t *data, size_t len, bool broad //Print int ESP_NOW_Serial_Class::availableForWrite() { //return ESP_NOW_MAX_DATA_LEN; - if (tx_ring_buf == NULL) { + if (tx_ring_buf == nullptr) { return 0; } return xRingbufferGetCurFreeSize(tx_ring_buf); @@ -178,7 +178,7 @@ size_t ESP_NOW_Serial_Class::tryToSend() { //_onSent will not be called anymore //the data is lost in this case vRingbufferReturnItem(tx_ring_buf, queued_buff); - queued_buff = NULL; + queued_buff = nullptr; xSemaphoreGive(tx_sem); end(); } @@ -188,12 +188,12 @@ size_t ESP_NOW_Serial_Class::tryToSend() { bool ESP_NOW_Serial_Class::checkForTxData() { //do we have something that failed the last time? resend_count = 0; - if (queued_buff == NULL) { + if (queued_buff == nullptr) { queued_buff = (uint8_t *)xRingbufferReceiveUpTo(tx_ring_buf, &queued_size, 0, ESP_NOW_MAX_DATA_LEN); } else { log_d(MACSTR " : PREVIOUS", MAC2STR(addr())); } - if (queued_buff != NULL) { + if (queued_buff != nullptr) { return tryToSend() > 0; } //log_d(MACSTR ": EMPTY", MAC2STR(addr())); @@ -203,7 +203,7 @@ bool ESP_NOW_Serial_Class::checkForTxData() { size_t ESP_NOW_Serial_Class::write(const uint8_t *buffer, size_t size, uint32_t timeout) { log_v(MACSTR ", size %u", MAC2STR(addr()), size); - if (tx_sem == NULL || tx_ring_buf == NULL || !added) { + if (tx_sem == nullptr || tx_ring_buf == nullptr || !added) { return 0; } size_t space = availableForWrite(); @@ -249,12 +249,12 @@ size_t ESP_NOW_Serial_Class::write(const uint8_t *buffer, size_t size, uint32_t //TX Done Callback void ESP_NOW_Serial_Class::onSent(bool success) { log_v(MACSTR " : %s", MAC2STR(addr()), success ? "OK" : "FAIL"); - if (tx_sem == NULL || tx_ring_buf == NULL || !added) { + if (tx_sem == nullptr || tx_ring_buf == nullptr || !added) { return; } if (success) { vRingbufferReturnItem(tx_ring_buf, queued_buff); - queued_buff = NULL; + queued_buff = nullptr; //send next packet? //log_d(MACSTR ": NEXT", MAC2STR(addr())); checkForTxData(); @@ -269,7 +269,7 @@ void ESP_NOW_Serial_Class::onSent(bool success) { //resend limit reached //the data is lost in this case vRingbufferReturnItem(tx_ring_buf, queued_buff); - queued_buff = NULL; + queued_buff = nullptr; log_e(MACSTR " : RE-SEND_MAX[%u]", MAC2STR(addr()), resend_count); //if we are not able to send the data and remove_on_fail is set, remove the peer if (_remove_on_fail) { diff --git a/libraries/ESP_NOW/src/ESP32_NOW_Serial.h b/libraries/ESP_NOW/src/ESP32_NOW_Serial.h index 7cc43d85ef8..5ccb7763ec2 100644 --- a/libraries/ESP_NOW/src/ESP32_NOW_Serial.h +++ b/libraries/ESP_NOW/src/ESP32_NOW_Serial.h @@ -28,7 +28,9 @@ class ESP_NOW_Serial_Class : public Stream, public ESP_NOW_Peer { size_t tryToSend(); public: - ESP_NOW_Serial_Class(const uint8_t *mac_addr, uint8_t channel, wifi_interface_t iface = WIFI_IF_AP, const uint8_t *lmk = NULL, bool remove_on_fail = false); + ESP_NOW_Serial_Class( + const uint8_t *mac_addr, uint8_t channel, wifi_interface_t iface = WIFI_IF_AP, const uint8_t *lmk = nullptr, bool remove_on_fail = false + ); ~ESP_NOW_Serial_Class(); size_t setRxBufferSize(size_t); size_t setTxBufferSize(size_t); From 95ae8cf9c63e3045034cdbc7b642d9bd0c5148b9 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Kuba=20Andr=C3=BDsek?= Date: Mon, 23 Jun 2025 14:02:47 +0200 Subject: [PATCH 065/173] feat(test): Enhance NVS test (#11481) * feat(test): Enhance NVS test * fix(nvs): Remove unused Unity header and improve Serial wait loop * refactor(nvs): Extract string increment logic into a separate function * refactor(test): Format long strings in expect_exact calls for better readability --- tests/validation/nvs/nvs.ino | 132 +++++++++++++++++++++++++++---- tests/validation/nvs/test_nvs.py | 24 ++++-- 2 files changed, 136 insertions(+), 20 deletions(-) diff --git a/tests/validation/nvs/nvs.ino b/tests/validation/nvs/nvs.ino index 20b5b460098..12c640c65b0 100644 --- a/tests/validation/nvs/nvs.ino +++ b/tests/validation/nvs/nvs.ino @@ -1,35 +1,139 @@ +#include #include +struct TestData { + uint8_t id; + uint16_t value; +}; + Preferences preferences; +void validate_types() { + assert(preferences.getType("char") == PT_I8); + assert(preferences.getType("uchar") == PT_U8); + assert(preferences.getType("short") == PT_I16); + assert(preferences.getType("ushort") == PT_U16); + assert(preferences.getType("int") == PT_I32); + assert(preferences.getType("uint") == PT_U32); + assert(preferences.getType("long") == PT_I32); + assert(preferences.getType("ulong") == PT_U32); + assert(preferences.getType("long64") == PT_I64); + assert(preferences.getType("ulong64") == PT_U64); + assert(preferences.getType("float") == PT_BLOB); + assert(preferences.getType("double") == PT_BLOB); + assert(preferences.getType("bool") == PT_U8); + assert(preferences.getType("str") == PT_STR); + assert(preferences.getType("strLen") == PT_STR); + assert(preferences.getType("struct") == PT_BLOB); +} + +// Function to increment string values +void incrementStringValues(String &val_string, char *val_string_buf, size_t buf_size) { + // Extract the number from string and increment it + val_string = "str" + String(val_string.substring(3).toInt() + 1); + + // Extract the number from strLen and increment it + String strLen_str = String(val_string_buf); + int strLen_num = strLen_str.substring(6).toInt(); + snprintf(val_string_buf, buf_size, "strLen%d", strLen_num + 1); +} + void setup() { Serial.begin(115200); - while (!Serial) { ; } preferences.begin("my-app", false); - // Get the counter value, if the key does not exist, return a default value of 0 - unsigned int counter = preferences.getUInt("counter", 0); + // Get the preferences value and if not exists, use default parameter + char val_char = preferences.getChar("char", 'A'); + unsigned char val_uchar = preferences.getUChar("uchar", 0); + int16_t val_short = preferences.getShort("short", 0); + uint16_t val_ushort = preferences.getUShort("ushort", 0); + int32_t val_int = preferences.getInt("int", 0); + uint32_t val_uint = preferences.getUInt("uint", 0); + int64_t val_long = preferences.getLong("long", 0); + uint32_t val_ulong = preferences.getULong("ulong", 0); + int64_t val_long64 = preferences.getLong64("long64", 0); + uint64_t val_ulong64 = preferences.getULong64("ulong64", 0); + float val_float = preferences.getFloat("float", 0.0f); + double val_double = preferences.getDouble("double", 0.0); + bool val_bool = preferences.getBool("bool", false); - // Print the counter to Serial Monitor - Serial.printf("Current counter value: %u\n", counter); + // Strings + String val_string = preferences.getString("str", "str0"); + char val_string_buf[20] = "strLen0"; + preferences.getString("strLen", val_string_buf, sizeof(val_string_buf)); - // Increase counter by 1 - counter++; + // Structure data + TestData test_data = {0, 0}; - // Store the counter to the Preferences - preferences.putUInt("counter", counter); + size_t struct_size = preferences.getBytes("struct", &test_data, sizeof(test_data)); + if (struct_size == 0) { + // First time - set initial values using parameter names + test_data.id = 1; + test_data.value = 100; + } - // Close the Preferences - preferences.end(); + Serial.printf("Values from Preferences: "); + Serial.printf("char: %c | uchar: %u | short: %d | ushort: %u | int: %ld | uint: %lu | ", val_char, val_uchar, val_short, val_ushort, val_int, val_uint); + Serial.printf("long: %lld | ulong: %lu | long64: %lld | ulong64: %llu | ", val_long, val_ulong, val_long64, val_ulong64); + Serial.printf( + "float: %.2f | double: %.2f | bool: %s | str: %s | strLen: %s | struct: {id:%u,val:%u}\n", val_float, val_double, val_bool ? "true" : "false", + val_string.c_str(), val_string_buf, test_data.id, test_data.value + ); - // Wait 1 second - delay(1000); + // Increment the values + val_char += 1; // Increment char A -> B + val_uchar += 1; + val_short += 1; + val_ushort += 1; + val_int += 1; + val_uint += 1; + val_long += 1; + val_ulong += 1; + val_long64 += 1; + val_ulong64 += 1; + val_float += 1.1f; + val_double += 1.1; + val_bool = !val_bool; // Toggle boolean value + + // Increment string values using function + incrementStringValues(val_string, val_string_buf, sizeof(val_string_buf)); + + test_data.id += 1; + test_data.value += 10; + + // Store the updated values back to Preferences + preferences.putChar("char", val_char); + preferences.putUChar("uchar", val_uchar); + preferences.putShort("short", val_short); + preferences.putUShort("ushort", val_ushort); + preferences.putInt("int", val_int); + preferences.putUInt("uint", val_uint); + preferences.putLong("long", val_long); + preferences.putULong("ulong", val_ulong); + preferences.putLong64("long64", val_long64); + preferences.putULong64("ulong64", val_ulong64); + preferences.putFloat("float", val_float); + preferences.putDouble("double", val_double); + preferences.putBool("bool", val_bool); + preferences.putString("str", val_string); + preferences.putString("strLen", val_string_buf); + preferences.putBytes("struct", &test_data, sizeof(test_data)); - // Restart ESP + // Check if the keys exist + assert(preferences.isKey("char")); + assert(preferences.isKey("struct")); + + // Validate the types of the keys + validate_types(); + + // Close the Preferences, wait and restart + preferences.end(); + Serial.flush(); + delay(1000); ESP.restart(); } diff --git a/tests/validation/nvs/test_nvs.py b/tests/validation/nvs/test_nvs.py index 424095a49ba..7f595b8e93d 100644 --- a/tests/validation/nvs/test_nvs.py +++ b/tests/validation/nvs/test_nvs.py @@ -4,11 +4,23 @@ def test_nvs(dut): LOGGER = logging.getLogger(__name__) - LOGGER.info("Expecting counter value 0") - dut.expect_exact("Current counter value: 0") + LOGGER.info("Expecting default values from Preferences") + dut.expect_exact( + "Values from Preferences: char: A | uchar: 0 | short: 0 | ushort: 0 | int: 0 | uint: 0 | long: 0 | ulong: 0 | " + "long64: 0 | ulong64: 0 | float: 0.00 | double: 0.00 | bool: false | str: str0 | strLen: strLen0 | " + "struct: {id:1,val:100}" + ) - LOGGER.info("Expecting counter value 1") - dut.expect_exact("Current counter value: 1") + LOGGER.info("Expecting updated preferences for the first time") + dut.expect_exact( + "Values from Preferences: char: B | uchar: 1 | short: 1 | ushort: 1 | int: 1 | uint: 1 | long: 1 | ulong: 1 | " + "long64: 1 | ulong64: 1 | float: 1.10 | double: 1.10 | bool: true | str: str1 | strLen: strLen1 | " + "struct: {id:2,val:110}" + ) - LOGGER.info("Expecting counter value 2") - dut.expect_exact("Current counter value: 2") + LOGGER.info("Expecting updated preferences for the second time") + dut.expect_exact( + "Values from Preferences: char: C | uchar: 2 | short: 2 | ushort: 2 | int: 2 | uint: 2 | long: 2 | ulong: 2 | " + "long64: 2 | ulong64: 2 | float: 2.20 | double: 2.20 | bool: false | str: str2 | strLen: strLen2 | " + "struct: {id:3,val:120}" + ) From bad975daa5910b42b48f9f1eda6154c45eb18e6f Mon Sep 17 00:00:00 2001 From: Sugar Glider Date: Tue, 24 Jun 2025 03:37:07 -0300 Subject: [PATCH 066/173] fix(uart): removes assert() to avoid reset (#11508) --- cores/esp32/esp32-hal-uart.c | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/cores/esp32/esp32-hal-uart.c b/cores/esp32/esp32-hal-uart.c index 2163e9b5f42..21fe88b57fb 100644 --- a/cores/esp32/esp32-hal-uart.c +++ b/cores/esp32/esp32-hal-uart.c @@ -305,7 +305,10 @@ static bool _uartTrySetIomuxPin(uart_port_t uart_num, int io_num, uint32_t idx) } // Assign the correct function to the GPIO. - assert(upin->iomux_func != -1); + if (upin->iomux_func == -1) { + log_e("IO#%d has bad IOMUX internal information. Switching to GPIO Matrix UART function.", io_num); + return false; + } if (uart_num < SOC_UART_HP_NUM) { gpio_iomux_out(io_num, upin->iomux_func, false); // If the pin is input, we also have to redirect the signal, in order to bypass the GPIO matrix. From 36d049659bcea13e4e2afb4463e7f29bc998ffa3 Mon Sep 17 00:00:00 2001 From: Me No Dev Date: Tue, 24 Jun 2025 12:30:29 +0300 Subject: [PATCH 067/173] IDF release/v5.5 (#11504) * IDF release/v5.5 4c3d086c * fix(prov): Enable BLE provisioning with NimBLE * fix(uart): idf 5.5 new gpio_iomux_* functions (#11507) * fix(uart): idf 5.5 new gpio_iomux_* functions * fix(uart): formatting and style * fix(uart): commentaries style fix * fix(uart): commentaries style fix * fix(uart): commentaries style fix * fix(uart): support to any idf 5.x version * fix(uart): changing assert in order to avoid reset * ci(pre-commit): Apply automatic fixes --------- Co-authored-by: pre-commit-ci-lite[bot] <117423508+pre-commit-ci-lite[bot]@users.noreply.github.com> * IDF release/v5.5 cbe9388f --------- Co-authored-by: Sugar Glider Co-authored-by: pre-commit-ci-lite[bot] <117423508+pre-commit-ci-lite[bot]@users.noreply.github.com> --- cores/esp32/esp32-hal-uart.c | 19 ++++-- .../WiFiProv/examples/WiFiProv/WiFiProv.ino | 2 +- libraries/WiFiProv/src/WiFiProv.cpp | 20 +++--- libraries/WiFiProv/src/WiFiProv.h | 4 +- package/package_esp32_index.template.json | 68 +++++++++---------- 5 files changed, 62 insertions(+), 51 deletions(-) diff --git a/cores/esp32/esp32-hal-uart.c b/cores/esp32/esp32-hal-uart.c index 3e33c588c3b..6e5a22da9f8 100644 --- a/cores/esp32/esp32-hal-uart.c +++ b/cores/esp32/esp32-hal-uart.c @@ -317,13 +317,24 @@ static bool _uartTrySetIomuxPin(uart_port_t uart_num, int io_num, uint32_t idx) } // Assign the correct function to the GPIO. - assert(upin->iomux_func != -1); + if (upin->iomux_func == -1) { + log_e("IO#%d has bad IOMUX internal information. Switching to GPIO Matrix UART function.", io_num); + return false; + } if (uart_num < SOC_UART_HP_NUM) { +#if ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 5, 0) + if (upin->input) { + gpio_iomux_input(io_num, upin->iomux_func, upin->signal); + } else { + gpio_iomux_output(io_num, upin->iomux_func); + } +#else gpio_iomux_out(io_num, upin->iomux_func, false); // If the pin is input, we also have to redirect the signal, in order to bypass the GPIO matrix. if (upin->input) { gpio_iomux_in(io_num, upin->signal); } +#endif } #if (SOC_UART_LP_NUM >= 1) && (SOC_RTCIO_PIN_COUNT >= 1) else { @@ -1276,11 +1287,11 @@ bool uartSetClockSource(uint8_t uartNum, uart_sclk_t clkSrc) { #if SOC_UART_LP_NUM >= 1 if (uart->num >= SOC_UART_HP_NUM) { switch (clkSrc) { - case UART_SCLK_XTAL: uart->_uart_clock_source = LP_UART_SCLK_XTAL_D2; break; + case UART_SCLK_XTAL: uart->_uart_clock_source = LP_UART_SCLK_XTAL_D2; break; #if CONFIG_IDF_TARGET_ESP32C5 - case UART_SCLK_RTC: uart->_uart_clock_source = LP_UART_SCLK_RC_FAST; break; + case UART_SCLK_RTC: uart->_uart_clock_source = LP_UART_SCLK_RC_FAST; break; #else - case UART_SCLK_RTC: uart->_uart_clock_source = LP_UART_SCLK_LP_FAST; break; + case UART_SCLK_RTC: uart->_uart_clock_source = LP_UART_SCLK_LP_FAST; break; #endif case UART_SCLK_DEFAULT: default: uart->_uart_clock_source = LP_UART_SCLK_DEFAULT; diff --git a/libraries/WiFiProv/examples/WiFiProv/WiFiProv.ino b/libraries/WiFiProv/examples/WiFiProv/WiFiProv.ino index 76025d75770..2f7d8c5089f 100644 --- a/libraries/WiFiProv/examples/WiFiProv/WiFiProv.ino +++ b/libraries/WiFiProv/examples/WiFiProv/WiFiProv.ino @@ -62,7 +62,7 @@ void setup() { WiFi.onEvent(SysProvEvent); // BLE Provisioning using the ESP SoftAP Prov works fine for any BLE SoC, including ESP32, ESP32S3 and ESP32C3. -#if CONFIG_BLUEDROID_ENABLED && !defined(USE_SOFT_AP) +#if (defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED)) && __has_include("esp_bt.h") && !defined(USE_SOFT_AP) Serial.println("Begin Provisioning using BLE"); // Sample uuid that user can pass during provisioning using BLE uint8_t uuid[16] = {0xb4, 0xdf, 0x5a, 0x1c, 0x3f, 0x6b, 0xf4, 0xbf, 0xea, 0x4a, 0x82, 0x03, 0x04, 0x90, 0x1a, 0x02}; diff --git a/libraries/WiFiProv/src/WiFiProv.cpp b/libraries/WiFiProv/src/WiFiProv.cpp index 31337196b5f..f372caf0e49 100644 --- a/libraries/WiFiProv/src/WiFiProv.cpp +++ b/libraries/WiFiProv/src/WiFiProv.cpp @@ -34,7 +34,7 @@ #endif #include -#if CONFIG_BLUEDROID_ENABLED +#if (defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED)) && __has_include("esp_bt.h") #include "network_provisioning/scheme_ble.h" #endif #include @@ -47,7 +47,7 @@ bool wifiLowLevelInit(bool persistent); -#if CONFIG_BLUEDROID_ENABLED +#if (defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED)) && __has_include("esp_bt.h") static const uint8_t custom_service_uuid[16] = { 0xb4, 0xdf, 0x5a, 0x1c, 0x3f, 0x6b, 0xf4, 0xbf, 0xea, 0x4a, 0x82, 0x03, 0x04, 0x90, 0x1a, 0x02, }; @@ -61,13 +61,13 @@ static void get_device_service_name(prov_scheme_t prov_scheme, char *service_nam log_e("esp_wifi_get_mac failed!"); return; } -#if CONFIG_IDF_TARGET_ESP32 && defined(CONFIG_BLUEDROID_ENABLED) +#if CONFIG_IDF_TARGET_ESP32 && (defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED)) && __has_include("esp_bt.h") if (prov_scheme == NETWORK_PROV_SCHEME_BLE) { snprintf(service_name, max, "%s%02X%02X%02X", SERV_NAME_PREFIX_PROV, eth_mac[3], eth_mac[4], eth_mac[5]); } else { #endif snprintf(service_name, max, "%s%02X%02X%02X", SERV_NAME_PREFIX_PROV, eth_mac[3], eth_mac[4], eth_mac[5]); -#if CONFIG_IDF_TARGET_ESP32 && defined(CONFIG_BLUEDROID_ENABLED) +#if CONFIG_IDF_TARGET_ESP32 && (defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED)) && __has_include("esp_bt.h") } #endif } @@ -78,20 +78,20 @@ void WiFiProvClass ::initProvision(prov_scheme_t prov_scheme, scheme_handler_t s return; } network_prov_mgr_config_t config; -#if CONFIG_BLUEDROID_ENABLED +#if (defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED)) && __has_include("esp_bt.h") if (prov_scheme == NETWORK_PROV_SCHEME_BLE) { config.scheme = network_prov_scheme_ble; } else { #endif config.scheme = network_prov_scheme_softap; -#if CONFIG_BLUEDROID_ENABLED +#if (defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED)) && __has_include("esp_bt.h") } if (scheme_handler == NETWORK_PROV_SCHEME_HANDLER_NONE) { #endif network_prov_event_handler_t scheme_event_handler = NETWORK_PROV_EVENT_HANDLER_NONE; memcpy(&config.scheme_event_handler, &scheme_event_handler, sizeof(network_prov_event_handler_t)); -#if CONFIG_BLUEDROID_ENABLED +#if (defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED)) && __has_include("esp_bt.h") } else if (scheme_handler == NETWORK_PROV_SCHEME_HANDLER_FREE_BTDM) { network_prov_event_handler_t scheme_event_handler = NETWORK_PROV_SCHEME_BLE_EVENT_HANDLER_FREE_BTDM; memcpy(&config.scheme_event_handler, &scheme_event_handler, sizeof(network_prov_event_handler_t)); @@ -133,7 +133,7 @@ void WiFiProvClass ::beginProvision( } static char service_name_temp[32]; if (provisioned == false) { -#if CONFIG_BLUEDROID_ENABLED +#if (defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED)) && __has_include("esp_bt.h") if (prov_scheme == NETWORK_PROV_SCHEME_BLE) { service_key = NULL; if (uuid == NULL) { @@ -148,7 +148,7 @@ void WiFiProvClass ::beginProvision( service_name = (const char *)service_name_temp; } -#if CONFIG_BLUEDROID_ENABLED +#if (defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED)) && __has_include("esp_bt.h") if (prov_scheme == NETWORK_PROV_SCHEME_BLE) { log_i("Starting AP using BLE. service_name : %s, pop : %s", service_name, pop); } else { @@ -158,7 +158,7 @@ void WiFiProvClass ::beginProvision( } else { log_i("Starting provisioning AP using SOFTAP. service_name : %s, password : %s, pop : %s", service_name, service_key, pop); } -#if CONFIG_BLUEDROID_ENABLED +#if (defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED)) && __has_include("esp_bt.h") } #endif if (network_prov_mgr_start_provisioning(security, pop, service_name, service_key) != ESP_OK) { diff --git a/libraries/WiFiProv/src/WiFiProv.h b/libraries/WiFiProv/src/WiFiProv.h index b660f8cf064..d34727b6896 100644 --- a/libraries/WiFiProv/src/WiFiProv.h +++ b/libraries/WiFiProv/src/WiFiProv.h @@ -29,7 +29,7 @@ //Select the scheme using which you want to provision typedef enum { NETWORK_PROV_SCHEME_SOFTAP, -#if CONFIG_BLUEDROID_ENABLED +#if (defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED)) && __has_include("esp_bt.h") NETWORK_PROV_SCHEME_BLE, #endif NETWORK_PROV_SCHEME_MAX @@ -37,7 +37,7 @@ typedef enum { typedef enum { NETWORK_PROV_SCHEME_HANDLER_NONE, -#if CONFIG_BLUEDROID_ENABLED +#if (defined(CONFIG_BLUEDROID_ENABLED) || defined(CONFIG_NIMBLE_ENABLED)) && __has_include("esp_bt.h") NETWORK_PROV_SCHEME_HANDLER_FREE_BTDM, NETWORK_PROV_SCHEME_HANDLER_FREE_BLE, NETWORK_PROV_SCHEME_HANDLER_FREE_BT, diff --git a/package/package_esp32_index.template.json b/package/package_esp32_index.template.json index 3ae2ff09ee6..1d753792b43 100644 --- a/package/package_esp32_index.template.json +++ b/package/package_esp32_index.template.json @@ -51,7 +51,7 @@ { "packager": "esp32", "name": "esp32-arduino-libs", - "version": "idf-release_v5.5-28ac0243-v1" + "version": "idf-release_v5.5-cbe9388f-v1" }, { "packager": "esp32", @@ -104,63 +104,63 @@ "tools": [ { "name": "esp32-arduino-libs", - "version": "idf-release_v5.5-28ac0243-v1", + "version": "idf-release_v5.5-cbe9388f-v1", "systems": [ { "host": "i686-mingw32", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-28ac0243-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-28ac0243-v1.zip", - "checksum": "SHA-256:280401ea803d8a782c11ef4f96cfbf80eb12a0f51bd12eac9cb96d6c26489f6e", - "size": "405149394" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-cbe9388f-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-cbe9388f-v1.zip", + "checksum": "SHA-256:b737ffb86a1b377db12dd610d06936ca8d85d877c872f532a68f6f0a3f666a3f", + "size": "421300036" }, { "host": "x86_64-mingw32", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-28ac0243-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-28ac0243-v1.zip", - "checksum": "SHA-256:280401ea803d8a782c11ef4f96cfbf80eb12a0f51bd12eac9cb96d6c26489f6e", - "size": "405149394" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-cbe9388f-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-cbe9388f-v1.zip", + "checksum": "SHA-256:b737ffb86a1b377db12dd610d06936ca8d85d877c872f532a68f6f0a3f666a3f", + "size": "421300036" }, { "host": "arm64-apple-darwin", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-28ac0243-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-28ac0243-v1.zip", - "checksum": "SHA-256:280401ea803d8a782c11ef4f96cfbf80eb12a0f51bd12eac9cb96d6c26489f6e", - "size": "405149394" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-cbe9388f-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-cbe9388f-v1.zip", + "checksum": "SHA-256:b737ffb86a1b377db12dd610d06936ca8d85d877c872f532a68f6f0a3f666a3f", + "size": "421300036" }, { "host": "x86_64-apple-darwin", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-28ac0243-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-28ac0243-v1.zip", - "checksum": "SHA-256:280401ea803d8a782c11ef4f96cfbf80eb12a0f51bd12eac9cb96d6c26489f6e", - "size": "405149394" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-cbe9388f-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-cbe9388f-v1.zip", + "checksum": "SHA-256:b737ffb86a1b377db12dd610d06936ca8d85d877c872f532a68f6f0a3f666a3f", + "size": "421300036" }, { "host": "x86_64-pc-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-28ac0243-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-28ac0243-v1.zip", - "checksum": "SHA-256:280401ea803d8a782c11ef4f96cfbf80eb12a0f51bd12eac9cb96d6c26489f6e", - "size": "405149394" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-cbe9388f-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-cbe9388f-v1.zip", + "checksum": "SHA-256:b737ffb86a1b377db12dd610d06936ca8d85d877c872f532a68f6f0a3f666a3f", + "size": "421300036" }, { "host": "i686-pc-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-28ac0243-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-28ac0243-v1.zip", - "checksum": "SHA-256:280401ea803d8a782c11ef4f96cfbf80eb12a0f51bd12eac9cb96d6c26489f6e", - "size": "405149394" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-cbe9388f-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-cbe9388f-v1.zip", + "checksum": "SHA-256:b737ffb86a1b377db12dd610d06936ca8d85d877c872f532a68f6f0a3f666a3f", + "size": "421300036" }, { "host": "aarch64-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-28ac0243-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-28ac0243-v1.zip", - "checksum": "SHA-256:280401ea803d8a782c11ef4f96cfbf80eb12a0f51bd12eac9cb96d6c26489f6e", - "size": "405149394" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-cbe9388f-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-cbe9388f-v1.zip", + "checksum": "SHA-256:b737ffb86a1b377db12dd610d06936ca8d85d877c872f532a68f6f0a3f666a3f", + "size": "421300036" }, { "host": "arm-linux-gnueabihf", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-28ac0243-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-28ac0243-v1.zip", - "checksum": "SHA-256:280401ea803d8a782c11ef4f96cfbf80eb12a0f51bd12eac9cb96d6c26489f6e", - "size": "405149394" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-cbe9388f-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-cbe9388f-v1.zip", + "checksum": "SHA-256:b737ffb86a1b377db12dd610d06936ca8d85d877c872f532a68f6f0a3f666a3f", + "size": "421300036" } ] }, From 882ef25a36399e3c32fc16986f00edddd3324d5a Mon Sep 17 00:00:00 2001 From: me-no-dev Date: Mon, 23 Jun 2025 12:11:32 +0300 Subject: [PATCH 068/173] fix(p4): Update hosted and wifi_remote components --- idf_component.yml | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/idf_component.yml b/idf_component.yml index 7f090cb26a0..450929a4067 100644 --- a/idf_component.yml +++ b/idf_component.yml @@ -105,11 +105,11 @@ dependencies: rules: - if: "target in [esp32s3]" espressif/esp_hosted: - version: "^0.0.25" + version: "^2.0.12" rules: - if: "target == esp32p4" espressif/esp_wifi_remote: - version: "^0.4.1" + version: "^0.13.0" rules: - if: "target == esp32p4" espressif/libsodium: From b7e5169ea17eabedb82171457f08f43c72bd2c5f Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jan=20Proch=C3=A1zka?= <90197375+P-R-O-C-H-Y@users.noreply.github.com> Date: Tue, 24 Jun 2025 13:43:05 +0200 Subject: [PATCH 069/173] fix(spi): Update spi bus for esp32s2 (#11510) --- cores/esp32/esp32-hal-spi.c | 32 ++++++++++++-------------------- cores/esp32/esp32-hal-spi.h | 12 +++--------- libraries/SD/src/SD.cpp | 4 +++- libraries/SPI/src/SPI.cpp | 1 + 4 files changed, 19 insertions(+), 30 deletions(-) diff --git a/cores/esp32/esp32-hal-spi.c b/cores/esp32/esp32-hal-spi.c index 6b8e3f8c013..d39aceb5f8d 100644 --- a/cores/esp32/esp32-hal-spi.c +++ b/cores/esp32/esp32-hal-spi.c @@ -79,16 +79,15 @@ struct spi_struct_t { #if CONFIG_IDF_TARGET_ESP32S2 // ESP32S2 -#define SPI_COUNT (3) +#define SPI_COUNT (2) -#define SPI_CLK_IDX(p) ((p == 0) ? SPICLK_OUT_MUX_IDX : ((p == 1) ? FSPICLK_OUT_MUX_IDX : ((p == 2) ? SPI3_CLK_OUT_MUX_IDX : 0))) -#define SPI_MISO_IDX(p) ((p == 0) ? SPIQ_OUT_IDX : ((p == 1) ? FSPIQ_OUT_IDX : ((p == 2) ? SPI3_Q_OUT_IDX : 0))) -#define SPI_MOSI_IDX(p) ((p == 0) ? SPID_IN_IDX : ((p == 1) ? FSPID_IN_IDX : ((p == 2) ? SPI3_D_IN_IDX : 0))) +#define SPI_CLK_IDX(p) ((p == 0) ? FSPICLK_OUT_MUX_IDX : ((p == 1) ? SPI3_CLK_OUT_MUX_IDX : 0)) +#define SPI_MISO_IDX(p) ((p == 0) ? FSPIQ_OUT_IDX : ((p == 1) ? SPI3_Q_OUT_IDX : 0)) +#define SPI_MOSI_IDX(p) ((p == 0) ? FSPID_IN_IDX : ((p == 1) ? SPI3_D_IN_IDX : 0)) -#define SPI_SPI_SS_IDX(n) ((n == 0) ? SPICS0_OUT_IDX : ((n == 1) ? SPICS1_OUT_IDX : 0)) -#define SPI_HSPI_SS_IDX(n) ((n == 0) ? SPI3_CS0_OUT_IDX : ((n == 1) ? SPI3_CS1_OUT_IDX : ((n == 2) ? SPI3_CS2_OUT_IDX : SPI3_CS0_OUT_IDX))) -#define SPI_FSPI_SS_IDX(n) ((n == 0) ? FSPICS0_OUT_IDX : ((n == 1) ? FSPICS1_OUT_IDX : ((n == 2) ? FSPICS2_OUT_IDX : FSPICS0_OUT_IDX))) -#define SPI_SS_IDX(p, n) ((p == 0) ? SPI_SPI_SS_IDX(n) : ((p == 1) ? SPI_SPI_SS_IDX(n) : ((p == 2) ? SPI_HSPI_SS_IDX(n) : 0))) +#define SPI_HSPI_SS_IDX(n) ((n == 0) ? SPI3_CS0_OUT_IDX : ((n == 1) ? SPI3_CS1_OUT_IDX : ((n == 2) ? SPI3_CS2_OUT_IDX : 0))) +#define SPI_FSPI_SS_IDX(n) ((n == 0) ? FSPICS0_OUT_IDX : ((n == 1) ? FSPICS1_OUT_IDX : ((n == 2) ? FSPICS2_OUT_IDX : 0))) +#define SPI_SS_IDX(p, n) ((p == 0) ? SPI_FSPI_SS_IDX(n) : ((p == 1) ? SPI_HSPI_SS_IDX(n) : 0)) #elif CONFIG_IDF_TARGET_ESP32S3 // ESP32S3 @@ -98,8 +97,8 @@ struct spi_struct_t { #define SPI_MISO_IDX(p) ((p == 0) ? FSPIQ_OUT_IDX : ((p == 1) ? SPI3_Q_OUT_IDX : 0)) #define SPI_MOSI_IDX(p) ((p == 0) ? FSPID_IN_IDX : ((p == 1) ? SPI3_D_IN_IDX : 0)) -#define SPI_HSPI_SS_IDX(n) ((n == 0) ? SPI3_CS0_OUT_IDX : ((n == 1) ? SPI3_CS1_OUT_IDX : 0)) -#define SPI_FSPI_SS_IDX(n) ((n == 0) ? FSPICS0_OUT_IDX : ((n == 1) ? FSPICS1_OUT_IDX : 0)) +#define SPI_HSPI_SS_IDX(n) ((n == 0) ? SPI3_CS0_OUT_IDX : ((n == 1) ? SPI3_CS1_OUT_IDX : ((n == 2) ? SPI3_CS2_OUT_IDX : 0))) +#define SPI_FSPI_SS_IDX(n) ((n == 0) ? FSPICS0_OUT_IDX : ((n == 1) ? FSPICS1_OUT_IDX : ((n == 2) ? FSPICS2_OUT_IDX : 0))) #define SPI_SS_IDX(p, n) ((p == 0) ? SPI_FSPI_SS_IDX(n) : ((p == 1) ? SPI_HSPI_SS_IDX(n) : 0)) #elif CONFIG_IDF_TARGET_ESP32P4 @@ -151,11 +150,7 @@ struct spi_struct_t { #define SPI_MUTEX_UNLOCK() // clang-format off static spi_t _spi_bus_array[] = { -#if CONFIG_IDF_TARGET_ESP32S2 - {(volatile spi_dev_t *)(DR_REG_SPI1_BASE), 0, -1, -1, -1, -1, false}, - {(volatile spi_dev_t *)(DR_REG_SPI2_BASE), 1, -1, -1, -1, -1, false}, - {(volatile spi_dev_t *)(DR_REG_SPI3_BASE), 2, -1, -1, -1, -1, false} -#elif CONFIG_IDF_TARGET_ESP32S3 || CONFIG_IDF_TARGET_ESP32P4 +#if CONFIG_IDF_TARGET_ESP32S2 ||CONFIG_IDF_TARGET_ESP32S3 || CONFIG_IDF_TARGET_ESP32P4 {(volatile spi_dev_t *)(DR_REG_SPI2_BASE), 0, -1, -1, -1, -1, false}, {(volatile spi_dev_t *)(DR_REG_SPI3_BASE), 1, -1, -1, -1, -1, false} #elif CONFIG_IDF_TARGET_ESP32C2 @@ -179,11 +174,7 @@ static spi_t _spi_bus_array[] = { #define SPI_MUTEX_UNLOCK() xSemaphoreGive(spi->lock) static spi_t _spi_bus_array[] = { -#if CONFIG_IDF_TARGET_ESP32S2 - {(volatile spi_dev_t *)(DR_REG_SPI1_BASE), NULL, 0, -1, -1, -1, -1, false}, - {(volatile spi_dev_t *)(DR_REG_SPI2_BASE), NULL, 1, -1, -1, -1, -1, false}, - {(volatile spi_dev_t *)(DR_REG_SPI3_BASE), NULL, 2, -1, -1, -1, -1, false} -#elif CONFIG_IDF_TARGET_ESP32S3 || CONFIG_IDF_TARGET_ESP32P4 +#if CONFIG_IDF_TARGET_ESP32S2 || CONFIG_IDF_TARGET_ESP32S3 || CONFIG_IDF_TARGET_ESP32P4 {(volatile spi_dev_t *)(DR_REG_SPI2_BASE), NULL, 0, -1, -1, -1, -1, false}, {(volatile spi_dev_t *)(DR_REG_SPI3_BASE), NULL, 1, -1, -1, -1, -1, false} #elif CONFIG_IDF_TARGET_ESP32C2 {(volatile spi_dev_t *)(DR_REG_SPI2_BASE), NULL, 0, -1, -1, -1, -1, false} @@ -621,6 +612,7 @@ void spiStopBus(spi_t *spi) { spi_t *spiStartBus(uint8_t spi_num, uint32_t clockDiv, uint8_t dataMode, uint8_t bitOrder) { if (spi_num >= SPI_COUNT) { + log_e("SPI bus index %d is out of range", spi_num); return NULL; } diff --git a/cores/esp32/esp32-hal-spi.h b/cores/esp32/esp32-hal-spi.h index 565ca43f60d..0284fea8829 100644 --- a/cores/esp32/esp32-hal-spi.h +++ b/cores/esp32/esp32-hal-spi.h @@ -27,19 +27,13 @@ extern "C" { #include #define SPI_HAS_TRANSACTION - -#ifdef CONFIG_IDF_TARGET_ESP32S2 -#define FSPI 1 //SPI 1 bus. ESP32S2: for external memory only (can use the same data lines but different SS) -#define HSPI 2 //SPI 2 bus. ESP32S2: external memory or device - it can be matrixed to any pins -#define SPI2 2 // Another name for ESP32S2 SPI 2 -#define SPI3 3 //SPI 3 bus. ESP32S2: device only - it can be matrixed to any pins -#elif CONFIG_IDF_TARGET_ESP32 +#ifdef CONFIG_IDF_TARGET_ESP32 #define FSPI 1 //SPI 1 bus attached to the flash (can use the same data lines but different SS) #define HSPI 2 //SPI 2 bus normally mapped to pins 12 - 15, but can be matrixed to any pins #define VSPI 3 //SPI 3 bus normally attached to pins 5, 18, 19 and 23, but can be matrixed to any pins #else -#define FSPI 0 // ESP32C2, C3, C6, H2, S3, P4 - SPI 2 bus -#define HSPI 1 // ESP32S3, P4 - SPI 3 bus +#define FSPI 0 // ESP32C2, C3, C6, H2, S2, S3, P4 - SPI 2 bus +#define HSPI 1 // ESP32S2, S3, P4 - SPI 3 bus #endif // This defines are not representing the real Divider of the ESP32 diff --git a/libraries/SD/src/SD.cpp b/libraries/SD/src/SD.cpp index eaec2483053..2d646276d87 100644 --- a/libraries/SD/src/SD.cpp +++ b/libraries/SD/src/SD.cpp @@ -27,7 +27,9 @@ bool SDFS::begin(uint8_t ssPin, SPIClass &spi, uint32_t frequency, const char *m return true; } - spi.begin(); + if (!spi.begin()) { + return false; + } _pdrv = sdcard_init(ssPin, &spi, frequency); if (_pdrv == 0xFF) { diff --git a/libraries/SPI/src/SPI.cpp b/libraries/SPI/src/SPI.cpp index c0d7665df03..7d8d44320a6 100644 --- a/libraries/SPI/src/SPI.cpp +++ b/libraries/SPI/src/SPI.cpp @@ -74,6 +74,7 @@ bool SPIClass::begin(int8_t sck, int8_t miso, int8_t mosi, int8_t ss) { _spi = spiStartBus(_spi_num, _div, SPI_MODE0, SPI_MSBFIRST); if (!_spi) { + log_e("SPI bus %d start failed.", _spi_num); return false; } From 30fb3cbb4f2eb749863f39a040601df3decd87a5 Mon Sep 17 00:00:00 2001 From: Lucas Saavedra Vaz <32426024+lucasssvaz@users.noreply.github.com> Date: Tue, 24 Jun 2025 08:43:27 -0300 Subject: [PATCH 070/173] fix(docs): Fix links and versions (#11505) * fix(docs): Fix links and versions * fix(docs): Apply suggestions and leftover substitutions --- docs/conf_common.py | 7 +++++++ docs/en/contributing.rst | 4 ++-- docs/en/esp-idf_component.rst | 10 ++++++---- docs/en/guides/docs_contributing.rst | 13 +++++-------- docs/en/index.rst | 1 + docs/en/lib_builder.rst | 16 ++++++++++------ docs/requirements.txt | 2 ++ 7 files changed, 33 insertions(+), 20 deletions(-) diff --git a/docs/conf_common.py b/docs/conf_common.py index 676cca899d5..6945c0d190d 100644 --- a/docs/conf_common.py +++ b/docs/conf_common.py @@ -2,6 +2,12 @@ from esp_docs.conf_docs import * # noqa: F403,F401 +# Used for substituting variables in the documentation +rst_prolog = """ +.. |version| replace:: 3.2.0 +.. |idf_version| replace:: 5.4 +""" + languages = ["en"] # idf_targets = [ @@ -27,6 +33,7 @@ extensions += [ # noqa: F405 "sphinx_copybutton", "sphinx_tabs.tabs", + "sphinx_substitution_extensions", # For allowing substitutions inside code blocks "esp_docs.esp_extensions.dummy_build_system", ] diff --git a/docs/en/contributing.rst b/docs/en/contributing.rst index 4ebe01cbf5b..7a7b99894eb 100644 --- a/docs/en/contributing.rst +++ b/docs/en/contributing.rst @@ -222,7 +222,7 @@ Documentation ------------- If you are contributing to the documentation, please follow the instructions described in the -`documentation guidelines `_ to properly format and test your changes. +`documentation guidelines `_ to properly format and test your changes. Testing and CI -------------- @@ -435,7 +435,7 @@ Documentation Checks ^^^^^^^^^^^^^^^^^^^^ The CI also checks the documentation for any compilation errors. This is important to ensure that the documentation layout is not broken. -To build the documentation locally, please refer to the `documentation guidelines `_. +To build the documentation locally, please refer to the `documentation guidelines `_. Code Style Checks ^^^^^^^^^^^^^^^^^ diff --git a/docs/en/esp-idf_component.rst b/docs/en/esp-idf_component.rst index f38dc44ec0c..13542cc3c87 100644 --- a/docs/en/esp-idf_component.rst +++ b/docs/en/esp-idf_component.rst @@ -14,9 +14,9 @@ For a simplified method, see `Installing using Boards Manager `_. -.. note:: Latest Arduino Core ESP32 version (3.0.X) is now compatible with `ESP-IDF v5.1 `_. Please consider this compatibility when using Arduino as a component in ESP-IDF. +.. note:: Latest Arduino Core ESP32 version (|version|) is now compatible with ESP-IDF v\ |idf_version|\ . Please consider this compatibility when using Arduino as a component in ESP-IDF. -For easiest use of Arduino framework as a ESP-IDF component, you can use the `IDF Component Manager `_ to add the Arduino component to your project. +For easiest use of Arduino framework as a ESP-IDF component, you can use the `IDF Component Manager `_ to add the Arduino component to your project. This will automatically clone the repository and its submodules. You can find the Arduino component in the `ESP Registry `_ together with dependencies list and examples. Installation @@ -32,14 +32,16 @@ Installing using IDF Component Manager To add the Arduino component to your project using the IDF Component Manager, run the following command in your project directory: .. code-block:: bash + :substitutions: - idf.py add-dependency "espressif/arduino-esp32^3.0.2" + idf.py add-dependency "espressif/arduino-esp32^|version|" Or you can start a new project from a template with the Arduino component: .. code-block:: bash + :substitutions: - idf.py create-project-from-example "espressif/arduino-esp32^3.0.2:hello_world" + idf.py create-project-from-example "espressif/arduino-esp32^|version|:hello_world" Manual installation of Arduino framework **************************************** diff --git a/docs/en/guides/docs_contributing.rst b/docs/en/guides/docs_contributing.rst index 20dc4c84cab..13f08c47618 100644 --- a/docs/en/guides/docs_contributing.rst +++ b/docs/en/guides/docs_contributing.rst @@ -49,11 +49,11 @@ Before starting your collaboration, you need to get the documentation source cod Requirements ************ -To properly work with the documentation, you need to install some packages in your system. +To build the documentation properly, you need to install some packages in your system. Note that depending on +your system, you may need to use a virtual environment to install the packages. .. code-block:: - pip install -U Sphinx pip install -r requirements.txt The requirements file is under the ``docs`` folder. @@ -62,17 +62,14 @@ Using Visual Studio Code ************************ If you are using the Visual Studio Code, you can install some extensions to help you while writing documentation. +For reStructuredText, you can install the `reStructuredText Pack `_ extension. -`reStructuredText Pack `_ - -We also recommend you install to grammar check extension to help you to review English grammar. - -`Grammarly `_ +We also recommend you to install some grammar check extension to help you to review English grammar. Building ******** -To build the documentation and generate the HTML files, you can use the following command inside the ``docs`` folder. After a successful build, you can check the files inside the `_build/en/generic/html` folder. +To build the documentation and generate the HTML files, you can use the following command inside the ``docs`` folder. After a successful build, you can check the files inside the ``_build/en/generic/html`` folder. .. code-block:: diff --git a/docs/en/index.rst b/docs/en/index.rst index 1314a8fc78d..89fc7e3bd6e 100644 --- a/docs/en/index.rst +++ b/docs/en/index.rst @@ -3,6 +3,7 @@ Welcome to ESP32 Arduino Core's documentation ############################################# Here you will find all the relevant information about the project. +This documentation is valid for the Arduino Core for ESP32 version |version| based on ESP-IDF |idf_version|. .. note:: This is a work in progress documentation and we will appreciate your help! We are looking for contributors! diff --git a/docs/en/lib_builder.rst b/docs/en/lib_builder.rst index e7edb331fd3..a8126c18edc 100644 --- a/docs/en/lib_builder.rst +++ b/docs/en/lib_builder.rst @@ -293,8 +293,9 @@ You have two options to run the Docker image to build the libraries. Manually or To run the Docker image manually, use the following command from the root of the ``arduino-esp32`` repository: .. code-block:: bash + :substitutions: - docker run --rm -it -v $PWD:/arduino-esp32 -e TERM=xterm-256color espressif/esp32-arduino-lib-builder:release-v5.1 + docker run --rm -it -v $PWD:/arduino-esp32 -e TERM=xterm-256color espressif/esp32-arduino-lib-builder:release-v|idf_version| This will start the Lib Builder UI for compiling the libraries. The above command explained: @@ -304,7 +305,7 @@ This will start the Lib Builder UI for compiling the libraries. The above comman - ``-t`` Allocate a pseudo-TTY; - ``-e TERM=xterm-256color``: Optional. Sets the terminal type to ``xterm-256color`` to display colors correctly; - ``-v $PWD:/arduino-esp32``: Optional. Mounts the current folder at ``/arduino-esp32`` inside the container. If not provided, the container will not copy the compiled libraries to the host machine; -- ``espressif/esp32-arduino-lib-builder:release-v5.1``: uses Docker image ``espressif/esp32-arduino-lib-builder`` with tag ``release-v5.1``. +- :substitution-code:`espressif/esp32-arduino-lib-builder:release-v|idf_version|`: uses Docker image ``espressif/esp32-arduino-lib-builder`` with tag :substitution-code:`release-v|idf_version|`. The ``latest`` tag is implicitly added by Docker when no tag is specified. It is recommended to use a specific version tag to ensure reproducibility of the build process. .. warning:: @@ -324,24 +325,27 @@ By default the docker container will run the user interface script. If you want For example, to run a terminal inside the container, you can run: .. code-block:: bash + :substitutions: - docker run -it espressif/esp32-arduino-lib-builder:release-v5.1 /bin/bash + docker run -it espressif/esp32-arduino-lib-builder:release-v|idf_version| /bin/bash Running the Docker image using the provided run script will depend on the host OS. Use the following command from the root of the ``arduino-esp32`` repository to execute the image in a Linux or macOS environment for -the ``release-v5.1`` tag: +the :substitution-code:`release-v|idf_version|` tag: .. code-block:: bash + :substitutions: - curl -LJO https://raw.githubusercontent.com/espressif/esp32-arduino-lib-builder/refs/heads/release/v5.1/tools/docker/run.sh + curl -LJO https://raw.githubusercontent.com/espressif/esp32-arduino-lib-builder/refs/heads/release/v|idf_version|/tools/docker/run.sh chmod +x run.sh ./run.sh $PWD For Windows, use the following command in PowerShell from the root of the ``arduino-esp32`` repository: .. code-block:: powershell + :substitutions: - Invoke-WebRequest -Uri "https://raw.githubusercontent.com/espressif/esp32-arduino-lib-builder/refs/heads/release/v5.1/tools/docker/run.ps1" -OutFile "run.ps1" + Invoke-WebRequest -Uri "https://raw.githubusercontent.com/espressif/esp32-arduino-lib-builder/refs/heads/release/v|idf_version|/tools/docker/run.ps1" -OutFile "run.ps1" .\run.ps1 $pwd As the script is unsigned, you may need to change the execution policy of the current session before running the script. diff --git a/docs/requirements.txt b/docs/requirements.txt index d3017fb5adc..ef2ab88cb65 100644 --- a/docs/requirements.txt +++ b/docs/requirements.txt @@ -1,5 +1,7 @@ +sphinx==4.5.0 esp-docs>=1.4.0 sphinx-copybutton==0.5.0 sphinx-tabs==3.2.0 numpydoc==1.5.0 standard-imghdr==3.13.0 +Sphinx-Substitution-Extensions==2022.2.16 From e9b0930f9dd1fb2e1df28c7156fa7892e4306b66 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jan=20Proch=C3=A1zka?= <90197375+P-R-O-C-H-Y@users.noreply.github.com> Date: Tue, 24 Jun 2025 14:34:17 +0200 Subject: [PATCH 071/173] ci(sizes): Update sizes workflow action --- .github/workflows/publishsizes.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/publishsizes.yml b/.github/workflows/publishsizes.yml index 8ff591e052b..f3c401d635c 100644 --- a/.github/workflows/publishsizes.yml +++ b/.github/workflows/publishsizes.yml @@ -66,7 +66,7 @@ jobs: path: ./artifacts/sizes-report/pr_num.txt - name: Report results - uses: P-R-O-C-H-Y/report-size-deltas@2043188c68f483a7b50527c4eacf609d05bb67a5 # sizes_v2 + uses: P-R-O-C-H-Y/report-size-deltas@sizes_v2 #2043188c68f483a7b50527c4eacf609d05bb67a5 # sizes_v2 with: sketches-reports-source: ${{ env.SKETCHES_REPORTS_PATH }} github-token: ${{ env.GITHUB_TOKEN }} From 0213200b3d66cae7849623dc6551503346228de3 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jan=20Proch=C3=A1zka?= <90197375+P-R-O-C-H-Y@users.noreply.github.com> Date: Tue, 24 Jun 2025 14:40:53 +0200 Subject: [PATCH 072/173] ci(sizes): Use commit id in report-size-delta action --- .github/workflows/publishsizes.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/publishsizes.yml b/.github/workflows/publishsizes.yml index f3c401d635c..fad2418668c 100644 --- a/.github/workflows/publishsizes.yml +++ b/.github/workflows/publishsizes.yml @@ -66,7 +66,7 @@ jobs: path: ./artifacts/sizes-report/pr_num.txt - name: Report results - uses: P-R-O-C-H-Y/report-size-deltas@sizes_v2 #2043188c68f483a7b50527c4eacf609d05bb67a5 # sizes_v2 + uses: P-R-O-C-H-Y/report-size-deltas@bea91d2c99ca80c88a883b39b1c4012f00ec3d09 # sizes_v2 with: sketches-reports-source: ${{ env.SKETCHES_REPORTS_PATH }} github-token: ${{ env.GITHUB_TOKEN }} From 9e61fa7e4bce59c05cb17c15b11b53b9bafca077 Mon Sep 17 00:00:00 2001 From: Me No Dev Date: Tue, 24 Jun 2025 17:23:50 +0300 Subject: [PATCH 073/173] IDF release/v5.4 (#11512) * IDF release/v5.4 f0f2980d * fix(p4): Allow custom pins on P4 for ESP-Hosted --- libraries/WiFi/src/WiFiGeneric.cpp | 12 +++- package/package_esp32_index.template.json | 68 +++++++++++------------ 2 files changed, 45 insertions(+), 35 deletions(-) diff --git a/libraries/WiFi/src/WiFiGeneric.cpp b/libraries/WiFi/src/WiFiGeneric.cpp index 3faf34fef34..bddcb5fd448 100644 --- a/libraries/WiFi/src/WiFiGeneric.cpp +++ b/libraries/WiFi/src/WiFiGeneric.cpp @@ -252,13 +252,23 @@ static bool wifiHostedInit() { if (!hosted_initialized) { hosted_initialized = true; struct esp_hosted_sdio_config conf = INIT_DEFAULT_HOST_SDIO_CONFIG(); +#ifdef BOARD_HAS_SDIO_ESP_HOSTED + conf.pin_clk.pin = BOARD_SDIO_ESP_HOSTED_CLK; + conf.pin_cmd.pin = BOARD_SDIO_ESP_HOSTED_CMD; + conf.pin_d0.pin = BOARD_SDIO_ESP_HOSTED_D0; + conf.pin_d1.pin = BOARD_SDIO_ESP_HOSTED_D1; + conf.pin_d2.pin = BOARD_SDIO_ESP_HOSTED_D2; + conf.pin_d3.pin = BOARD_SDIO_ESP_HOSTED_D3; + conf.pin_reset.pin = BOARD_SDIO_ESP_HOSTED_RESET; +#else conf.pin_clk.pin = CONFIG_ESP_SDIO_PIN_CLK; conf.pin_cmd.pin = CONFIG_ESP_SDIO_PIN_CMD; conf.pin_d0.pin = CONFIG_ESP_SDIO_PIN_D0; conf.pin_d1.pin = CONFIG_ESP_SDIO_PIN_D1; conf.pin_d2.pin = CONFIG_ESP_SDIO_PIN_D2; conf.pin_d3.pin = CONFIG_ESP_SDIO_PIN_D3; - //conf.pin_rst.pin = CONFIG_ESP_SDIO_GPIO_RESET_SLAVE; + conf.pin_reset.pin = CONFIG_ESP_SDIO_GPIO_RESET_SLAVE; +#endif // esp_hosted_sdio_set_config() will fail on second attempt but here temporarily to not cause exception on reinit if (esp_hosted_sdio_set_config(&conf) != ESP_OK || esp_hosted_init() != ESP_OK) { log_e("esp_hosted_init failed!"); diff --git a/package/package_esp32_index.template.json b/package/package_esp32_index.template.json index 22d3cc05455..9c1516dbacd 100644 --- a/package/package_esp32_index.template.json +++ b/package/package_esp32_index.template.json @@ -51,7 +51,7 @@ { "packager": "esp32", "name": "esp32-arduino-libs", - "version": "idf-release_v5.4-aed8bdc8-v1" + "version": "idf-release_v5.4-f0f2980d-v1" }, { "packager": "esp32", @@ -104,63 +104,63 @@ "tools": [ { "name": "esp32-arduino-libs", - "version": "idf-release_v5.4-aed8bdc8-v1", + "version": "idf-release_v5.4-f0f2980d-v1", "systems": [ { "host": "i686-mingw32", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.4/esp32-arduino-libs-idf-release_v5.4-aed8bdc8-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.4-aed8bdc8-v1.zip", - "checksum": "SHA-256:448691c3171f79b2136e4ab8006e9c78bd1627156dab1365fff8f8867a6a7e5b", - "size": "353758763" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.4/esp32-arduino-libs-idf-release_v5.4-f0f2980d-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.4-f0f2980d-v1.zip", + "checksum": "SHA-256:b1a77c83634111d78a356f1d1756dc815eeeb91605c5d8714a0db7cccbd0bede", + "size": "353978504" }, { "host": "x86_64-mingw32", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.4/esp32-arduino-libs-idf-release_v5.4-aed8bdc8-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.4-aed8bdc8-v1.zip", - "checksum": "SHA-256:448691c3171f79b2136e4ab8006e9c78bd1627156dab1365fff8f8867a6a7e5b", - "size": "353758763" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.4/esp32-arduino-libs-idf-release_v5.4-f0f2980d-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.4-f0f2980d-v1.zip", + "checksum": "SHA-256:b1a77c83634111d78a356f1d1756dc815eeeb91605c5d8714a0db7cccbd0bede", + "size": "353978504" }, { "host": "arm64-apple-darwin", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.4/esp32-arduino-libs-idf-release_v5.4-aed8bdc8-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.4-aed8bdc8-v1.zip", - "checksum": "SHA-256:448691c3171f79b2136e4ab8006e9c78bd1627156dab1365fff8f8867a6a7e5b", - "size": "353758763" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.4/esp32-arduino-libs-idf-release_v5.4-f0f2980d-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.4-f0f2980d-v1.zip", + "checksum": "SHA-256:b1a77c83634111d78a356f1d1756dc815eeeb91605c5d8714a0db7cccbd0bede", + "size": "353978504" }, { "host": "x86_64-apple-darwin", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.4/esp32-arduino-libs-idf-release_v5.4-aed8bdc8-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.4-aed8bdc8-v1.zip", - "checksum": "SHA-256:448691c3171f79b2136e4ab8006e9c78bd1627156dab1365fff8f8867a6a7e5b", - "size": "353758763" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.4/esp32-arduino-libs-idf-release_v5.4-f0f2980d-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.4-f0f2980d-v1.zip", + "checksum": "SHA-256:b1a77c83634111d78a356f1d1756dc815eeeb91605c5d8714a0db7cccbd0bede", + "size": "353978504" }, { "host": "x86_64-pc-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.4/esp32-arduino-libs-idf-release_v5.4-aed8bdc8-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.4-aed8bdc8-v1.zip", - "checksum": "SHA-256:448691c3171f79b2136e4ab8006e9c78bd1627156dab1365fff8f8867a6a7e5b", - "size": "353758763" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.4/esp32-arduino-libs-idf-release_v5.4-f0f2980d-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.4-f0f2980d-v1.zip", + "checksum": "SHA-256:b1a77c83634111d78a356f1d1756dc815eeeb91605c5d8714a0db7cccbd0bede", + "size": "353978504" }, { "host": "i686-pc-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.4/esp32-arduino-libs-idf-release_v5.4-aed8bdc8-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.4-aed8bdc8-v1.zip", - "checksum": "SHA-256:448691c3171f79b2136e4ab8006e9c78bd1627156dab1365fff8f8867a6a7e5b", - "size": "353758763" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.4/esp32-arduino-libs-idf-release_v5.4-f0f2980d-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.4-f0f2980d-v1.zip", + "checksum": "SHA-256:b1a77c83634111d78a356f1d1756dc815eeeb91605c5d8714a0db7cccbd0bede", + "size": "353978504" }, { "host": "aarch64-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.4/esp32-arduino-libs-idf-release_v5.4-aed8bdc8-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.4-aed8bdc8-v1.zip", - "checksum": "SHA-256:448691c3171f79b2136e4ab8006e9c78bd1627156dab1365fff8f8867a6a7e5b", - "size": "353758763" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.4/esp32-arduino-libs-idf-release_v5.4-f0f2980d-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.4-f0f2980d-v1.zip", + "checksum": "SHA-256:b1a77c83634111d78a356f1d1756dc815eeeb91605c5d8714a0db7cccbd0bede", + "size": "353978504" }, { "host": "arm-linux-gnueabihf", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.4/esp32-arduino-libs-idf-release_v5.4-aed8bdc8-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.4-aed8bdc8-v1.zip", - "checksum": "SHA-256:448691c3171f79b2136e4ab8006e9c78bd1627156dab1365fff8f8867a6a7e5b", - "size": "353758763" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.4/esp32-arduino-libs-idf-release_v5.4-f0f2980d-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.4-f0f2980d-v1.zip", + "checksum": "SHA-256:b1a77c83634111d78a356f1d1756dc815eeeb91605c5d8714a0db7cccbd0bede", + "size": "353978504" } ] }, From 21640ac82a1bb5efa8cf0b3841be1ac80add6785 Mon Sep 17 00:00:00 2001 From: Lucas Saavedra Vaz <32426024+lucasssvaz@users.noreply.github.com> Date: Thu, 26 Jun 2025 12:43:50 +0300 Subject: [PATCH 074/173] fix(webserver): Validate header inputs --- libraries/WebServer/src/WebServer.cpp | 10 ++++++++++ 1 file changed, 10 insertions(+) diff --git a/libraries/WebServer/src/WebServer.cpp b/libraries/WebServer/src/WebServer.cpp index 652a86f587f..7523e40259b 100644 --- a/libraries/WebServer/src/WebServer.cpp +++ b/libraries/WebServer/src/WebServer.cpp @@ -502,6 +502,16 @@ void WebServer::stop() { } void WebServer::sendHeader(const String &name, const String &value, bool first) { + if (name.indexOf('\r') != -1 || name.indexOf('\n') != -1) { + log_e("Invalid character in HTTP header name"); + return; + } + + if (value.indexOf('\r') != -1 || value.indexOf('\n') != -1) { + log_e("Invalid character in HTTP header value"); + return; + } + RequestArgument *header = new RequestArgument(); header->key = name; header->value = value; From 875b92303510dca333620aa88140a993275d28fd Mon Sep 17 00:00:00 2001 From: Lucas Saavedra Vaz <32426024+lucasssvaz@users.noreply.github.com> Date: Mon, 30 Jun 2025 05:50:17 -0300 Subject: [PATCH 075/173] ci(ext_lib): Skip P4 in ArduinoBLE test (#11520) --- .github/workflows/lib.json | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/.github/workflows/lib.json b/.github/workflows/lib.json index 5b93d6689ef..f1ff111d25b 100644 --- a/.github/workflows/lib.json +++ b/.github/workflows/lib.json @@ -9,7 +9,8 @@ { "name": "ArduinoBLE", "exclude_targets": [ - "esp32s2" + "esp32s2", + "esp32p4" ], "sketch_path": [ "~/Arduino/libraries/ArduinoBLE/examples/Central/Scan/Scan.ino" From 9a35d9455f23f0f7055fbc6afb6845979f27cb04 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Kuba=20Andr=C3=BDsek?= Date: Mon, 30 Jun 2025 10:50:49 +0200 Subject: [PATCH 076/173] feat(SDFS): Add destructor for SD card to clean up resources (#11521) * feat(test): Enhance NVS test * fix(nvs): Remove unused Unity header and improve Serial wait loop * refactor(nvs): Extract string increment logic into a separate function * refactor(test): Format long strings in expect_exact calls for better readability * feat(SDFS): Add destructor to clean up resources --- libraries/SD/src/SD.cpp | 4 ++++ libraries/SD/src/SD.h | 1 + 2 files changed, 5 insertions(+) diff --git a/libraries/SD/src/SD.cpp b/libraries/SD/src/SD.cpp index 2d646276d87..077a7c1121f 100644 --- a/libraries/SD/src/SD.cpp +++ b/libraries/SD/src/SD.cpp @@ -22,6 +22,10 @@ using namespace fs; SDFS::SDFS(FSImplPtr impl) : FS(impl), _pdrv(0xFF) {} +SDFS::~SDFS() { + end(); +} + bool SDFS::begin(uint8_t ssPin, SPIClass &spi, uint32_t frequency, const char *mountpoint, uint8_t max_files, bool format_if_empty) { if (_pdrv != 0xFF) { return true; diff --git a/libraries/SD/src/SD.h b/libraries/SD/src/SD.h index aebc781a5e3..d8252ee44f7 100644 --- a/libraries/SD/src/SD.h +++ b/libraries/SD/src/SD.h @@ -26,6 +26,7 @@ class SDFS : public FS { public: SDFS(FSImplPtr impl); + ~SDFS(); bool begin( uint8_t ssPin = SS, SPIClass &spi = SPI, uint32_t frequency = 4000000, const char *mountpoint = "/sd", uint8_t max_files = 5, bool format_if_empty = false ); From 6754b1962c791de6c0a06120c42ad7f248600703 Mon Sep 17 00:00:00 2001 From: Lucas Saavedra Vaz <32426024+lucasssvaz@users.noreply.github.com> Date: Mon, 30 Jun 2025 06:02:12 -0300 Subject: [PATCH 077/173] feat(esp_now): Add support for ESP NOW V2 (#11524) * feat(esp_now): Add support for ESP NOW V2 * fix(esp_now): Return -1 on error --- .../ESP_NOW_Broadcast_Master.ino | 2 + .../ESP_NOW_Broadcast_Slave.ino | 2 + .../ESP_NOW_Network/ESP_NOW_Network.ino | 9 ++- .../ESP_NOW_Serial/ESP_NOW_Serial.ino | 1 + libraries/ESP_NOW/src/ESP32_NOW.cpp | 65 ++++++++++++++++--- libraries/ESP_NOW/src/ESP32_NOW.h | 10 ++- libraries/ESP_NOW/src/ESP32_NOW_Serial.cpp | 24 +++++-- 7 files changed, 98 insertions(+), 15 deletions(-) diff --git a/libraries/ESP_NOW/examples/ESP_NOW_Broadcast_Master/ESP_NOW_Broadcast_Master.ino b/libraries/ESP_NOW/examples/ESP_NOW_Broadcast_Master/ESP_NOW_Broadcast_Master.ino index 025a53c913b..7b71c0e432d 100644 --- a/libraries/ESP_NOW/examples/ESP_NOW_Broadcast_Master/ESP_NOW_Broadcast_Master.ino +++ b/libraries/ESP_NOW/examples/ESP_NOW_Broadcast_Master/ESP_NOW_Broadcast_Master.ino @@ -86,6 +86,8 @@ void setup() { ESP.restart(); } + Serial.printf("ESP-NOW version: %d, max data length: %d\n", ESP_NOW.getVersion(), ESP_NOW.getMaxDataLen()); + Serial.println("Setup complete. Broadcasting messages every 5 seconds."); } diff --git a/libraries/ESP_NOW/examples/ESP_NOW_Broadcast_Slave/ESP_NOW_Broadcast_Slave.ino b/libraries/ESP_NOW/examples/ESP_NOW_Broadcast_Slave/ESP_NOW_Broadcast_Slave.ino index e61524b64f9..ca0ece2fff8 100644 --- a/libraries/ESP_NOW/examples/ESP_NOW_Broadcast_Slave/ESP_NOW_Broadcast_Slave.ino +++ b/libraries/ESP_NOW/examples/ESP_NOW_Broadcast_Slave/ESP_NOW_Broadcast_Slave.ino @@ -104,6 +104,8 @@ void setup() { ESP.restart(); } + Serial.printf("ESP-NOW version: %d, max data length: %d\n", ESP_NOW.getVersion(), ESP_NOW.getMaxDataLen()); + // Register the new peer callback ESP_NOW.onNewPeer(register_new_master, nullptr); diff --git a/libraries/ESP_NOW/examples/ESP_NOW_Network/ESP_NOW_Network.ino b/libraries/ESP_NOW/examples/ESP_NOW_Network/ESP_NOW_Network.ino index 6731340c922..d30d6cd40cf 100644 --- a/libraries/ESP_NOW/examples/ESP_NOW_Network/ESP_NOW_Network.ino +++ b/libraries/ESP_NOW/examples/ESP_NOW_Network/ESP_NOW_Network.ino @@ -75,7 +75,12 @@ // The following struct is used to send data to the peer device. // We use the attribute "packed" to ensure that the struct is not padded (all data // is contiguous in the memory and without gaps). -// The maximum size of the complete message is 250 bytes (ESP_NOW_MAX_DATA_LEN). +// The maximum size of the payload is 250 bytes (ESP_NOW_MAX_DATA_LEN) for ESP-NOW v1.0. +// For ESP-NOW v2.0, the maximum size of the payload is 1470 bytes (ESP_NOW_MAX_DATA_LEN_V2). +// You can use ESP_NOW.getMaxDataLen() after calling ESP_NOW.begin() to get the maximum size +// of the data that can be sent. +// Read about the compatibility between ESP-NOW v1.0 and v2.0 in the ESP-IDF documentation: +// https://docs.espressif.com/projects/esp-idf/en/latest/esp32/api-reference/network/esp_now.html#frame-format typedef struct { uint32_t count; @@ -276,6 +281,8 @@ void setup() { fail_reboot(); } + Serial.printf("ESP-NOW version: %d, max data length: %d\n", ESP_NOW.getVersion(), ESP_NOW.getMaxDataLen()); + if (!broadcast_peer.begin()) { Serial.println("Failed to initialize broadcast peer"); fail_reboot(); diff --git a/libraries/ESP_NOW/examples/ESP_NOW_Serial/ESP_NOW_Serial.ino b/libraries/ESP_NOW/examples/ESP_NOW_Serial/ESP_NOW_Serial.ino index e1f28382fe2..2a5406cd844 100644 --- a/libraries/ESP_NOW/examples/ESP_NOW_Serial/ESP_NOW_Serial.ino +++ b/libraries/ESP_NOW/examples/ESP_NOW_Serial/ESP_NOW_Serial.ino @@ -64,6 +64,7 @@ void setup() { // Start the ESP-NOW communication Serial.println("ESP-NOW communication starting..."); NowSerial.begin(115200); + Serial.printf("ESP-NOW version: %d, max data length: %d\n", ESP_NOW.getVersion(), ESP_NOW.getMaxDataLen()); Serial.println("You can now send data to the peer device using the Serial Monitor.\n"); } diff --git a/libraries/ESP_NOW/src/ESP32_NOW.cpp b/libraries/ESP_NOW/src/ESP32_NOW.cpp index 25a5609e8db..b5cdc3f3da7 100644 --- a/libraries/ESP_NOW/src/ESP32_NOW.cpp +++ b/libraries/ESP_NOW/src/ESP32_NOW.cpp @@ -140,7 +140,10 @@ static void _esp_now_tx_cb(const uint8_t *mac_addr, esp_now_send_status_t status } } -ESP_NOW_Class::ESP_NOW_Class() {} +ESP_NOW_Class::ESP_NOW_Class() { + max_data_len = 0; + version = 0; +} ESP_NOW_Class::~ESP_NOW_Class() {} @@ -155,6 +158,23 @@ bool ESP_NOW_Class::begin(const uint8_t *pmk) { return false; } + // Unfortunately we can't get the ESP-NOW version before initializing the Wi-Fi + uint32_t esp_now_version; + err = esp_now_get_version(&esp_now_version); + if (err != ESP_OK) { + log_w("esp_now_get_version failed! Assuming ESP-NOW v1.0"); + esp_now_version = 1; + } + + if (esp_now_version == 1) { + max_data_len = ESP_NOW_MAX_DATA_LEN; + } else { + max_data_len = ESP_NOW_MAX_DATA_LEN_V2; + } + + version = esp_now_version; + log_i("ESP-NOW version: %lu, max_data_len: %lu", version, max_data_len); + _esp_now_has_begun = true; memset(_esp_now_peers, 0, sizeof(ESP_NOW_Peer *) * ESP_NOW_MAX_TOTAL_PEER_NUM); @@ -212,7 +232,7 @@ bool ESP_NOW_Class::end() { return true; } -int ESP_NOW_Class::getTotalPeerCount() { +int ESP_NOW_Class::getTotalPeerCount() const { if (!_esp_now_has_begun) { return -1; } @@ -225,7 +245,7 @@ int ESP_NOW_Class::getTotalPeerCount() { return num.total_num; } -int ESP_NOW_Class::getEncryptedPeerCount() { +int ESP_NOW_Class::getEncryptedPeerCount() const { if (!_esp_now_has_begun) { return -1; } @@ -238,16 +258,38 @@ int ESP_NOW_Class::getEncryptedPeerCount() { return num.encrypt_num; } +int ESP_NOW_Class::getMaxDataLen() const { + if (max_data_len == 0) { + log_e("ESP-NOW not initialized. Please call begin() first to get the max data length."); + return -1; + } + + return max_data_len; +} + +int ESP_NOW_Class::getVersion() const { + if (version == 0) { + log_e("ESP-NOW not initialized. Please call begin() first to get the version."); + return -1; + } + + return version; +} + int ESP_NOW_Class::availableForWrite() { - return ESP_NOW_MAX_DATA_LEN; + int available = getMaxDataLen(); + if (available < 0) { + return 0; + } + return available; } size_t ESP_NOW_Class::write(const uint8_t *data, size_t len) { if (!_esp_now_has_begun) { return 0; } - if (len > ESP_NOW_MAX_DATA_LEN) { - len = ESP_NOW_MAX_DATA_LEN; + if (len > max_data_len) { + len = max_data_len; } esp_err_t result = esp_now_send(nullptr, data, len); if (result == ESP_OK) { @@ -386,8 +428,15 @@ size_t ESP_NOW_Peer::send(const uint8_t *data, int len) { log_e("Peer not added."); return 0; } - if (len > ESP_NOW_MAX_DATA_LEN) { - len = ESP_NOW_MAX_DATA_LEN; + + int max_data_len = ESP_NOW.getMaxDataLen(); + if (max_data_len < 0) { + log_e("Error getting max data length."); + return 0; + } + + if (len > max_data_len) { + len = max_data_len; } esp_err_t result = esp_now_send(mac, data, len); if (result == ESP_OK) { diff --git a/libraries/ESP_NOW/src/ESP32_NOW.h b/libraries/ESP_NOW/src/ESP32_NOW.h index 5940cfa2221..5b5bbe72673 100644 --- a/libraries/ESP_NOW/src/ESP32_NOW.h +++ b/libraries/ESP_NOW/src/ESP32_NOW.h @@ -23,8 +23,10 @@ class ESP_NOW_Class : public Print { bool begin(const uint8_t *pmk = nullptr /* 16 bytes */); bool end(); - int getTotalPeerCount(); - int getEncryptedPeerCount(); + int getTotalPeerCount() const; + int getEncryptedPeerCount() const; + int getMaxDataLen() const; + int getVersion() const; int availableForWrite(); size_t write(const uint8_t *data, size_t len); @@ -34,6 +36,10 @@ class ESP_NOW_Class : public Print { void onNewPeer(void (*cb)(const esp_now_recv_info_t *info, const uint8_t *data, int len, void *arg), void *arg); bool removePeer(ESP_NOW_Peer &peer); + +protected: + size_t max_data_len; + uint32_t version; }; class ESP_NOW_Peer { diff --git a/libraries/ESP_NOW/src/ESP32_NOW_Serial.cpp b/libraries/ESP_NOW/src/ESP32_NOW_Serial.cpp index edd6e32aacc..c86c8546c5b 100644 --- a/libraries/ESP_NOW/src/ESP32_NOW_Serial.cpp +++ b/libraries/ESP_NOW/src/ESP32_NOW_Serial.cpp @@ -70,8 +70,25 @@ bool ESP_NOW_Serial_Class::begin(unsigned long baud) { //xSemaphoreTake(tx_sem, 0); xSemaphoreGive(tx_sem); } - setRxBufferSize(1024); //default if not preset - setTxBufferSize(1024); //default if not preset + + size_t buf_size = 0; + if (ESP_NOW.getVersion() == 2) { + // ESP-NOW v2.0 has a larger maximum data length, so we need to increase the buffer sizes + // to hold around 3-4 packets + buf_size = setRxBufferSize(4096); + buf_size &= setTxBufferSize(4096); + } else { + // ESP-NOW v1.0 has a smaller maximum data length, so we can use the default buffer sizes + // to hold around 3-4 packets + buf_size = setRxBufferSize(1024); + buf_size &= setTxBufferSize(1024); + } + + if (buf_size == 0) { + log_e("Failed to set buffer size"); + return false; + } + return true; } @@ -164,7 +181,6 @@ void ESP_NOW_Serial_Class::onReceive(const uint8_t *data, size_t len, bool broad //Print int ESP_NOW_Serial_Class::availableForWrite() { - //return ESP_NOW_MAX_DATA_LEN; if (tx_ring_buf == nullptr) { return 0; } @@ -189,7 +205,7 @@ bool ESP_NOW_Serial_Class::checkForTxData() { //do we have something that failed the last time? resend_count = 0; if (queued_buff == nullptr) { - queued_buff = (uint8_t *)xRingbufferReceiveUpTo(tx_ring_buf, &queued_size, 0, ESP_NOW_MAX_DATA_LEN); + queued_buff = (uint8_t *)xRingbufferReceiveUpTo(tx_ring_buf, &queued_size, 0, ESP_NOW.getMaxDataLen()); } else { log_d(MACSTR " : PREVIOUS", MAC2STR(addr())); } From 6476260e8fc8759e96a9c4cbff499436e421b2cc Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Mattias=20Sch=C3=A4ffersmann?= <117294155+MattiasTF@users.noreply.github.com> Date: Mon, 30 Jun 2025 12:04:11 +0200 Subject: [PATCH 078/173] fix(esp32): Fix appending to Strings longer than 64k (#11523) If oldLen is truncated to uint16_t, appending to a String that is longer than 65535 bytes will create a broken string. --- cores/esp32/WString.cpp | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/cores/esp32/WString.cpp b/cores/esp32/WString.cpp index 18e64767545..6632aa5f85d 100644 --- a/cores/esp32/WString.cpp +++ b/cores/esp32/WString.cpp @@ -180,7 +180,7 @@ bool String::changeBuffer(unsigned int maxStrLen) { if (maxStrLen < sizeof(sso.buff) - 1) { if (isSSO() || !buffer()) { // Already using SSO, nothing to do - uint16_t oldLen = len(); + size_t oldLen = len(); setSSO(true); setLen(oldLen); } else { // if bufptr && !isSSO() @@ -188,7 +188,7 @@ bool String::changeBuffer(unsigned int maxStrLen) { char temp[sizeof(sso.buff)]; memcpy(temp, buffer(), maxStrLen); free(wbuffer()); - uint16_t oldLen = len(); + size_t oldLen = len(); setSSO(true); memcpy(wbuffer(), temp, maxStrLen); setLen(oldLen); @@ -201,7 +201,7 @@ bool String::changeBuffer(unsigned int maxStrLen) { if (newSize > CAPACITY_MAX) { return false; } - uint16_t oldLen = len(); + size_t oldLen = len(); char *newbuffer = (char *)realloc(isSSO() ? nullptr : wbuffer(), newSize); if (newbuffer) { size_t oldSize = capacity() + 1; // include NULL. From 6e60f2f6b9e28ac20082e7e7387eaa90f362da48 Mon Sep 17 00:00:00 2001 From: LusimiCollado Date: Mon, 30 Jun 2025 12:18:02 +0200 Subject: [PATCH 079/173] =?UTF-8?q?feat(variant):=20add=20kode=20dot=20ESP?= =?UTF-8?q?32-S3=20board=20with=20QSPI=20LCD,=20SD=20and=20GPIO=20?= =?UTF-8?q?=E2=80=A6=20(#11371)?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit * feat(variant): add KodeDot ESP32-S3 board with QSPI LCD, SD and GPIO header * fix(kodedot): Reorder board definitions and translate comments to English * fix(kodedot): Clean up OTA override and remove unused partition menu for kode dot * fix(kodedot): Build board changed from ESP32S3_DEV to KODE_DOT on kode dot board * ci(pre-commit): Fix whitespace, EOLs and codespell 'Analog' * ci(pre-commit): Add bash script formatter * fix(merge): New name and description for custom merge tool and change partitions table to variants folder --- boards.txt | 57 ++++++++++++ platform.txt | 16 ++++ variants/kodedot/custom_ota_override.cpp | 14 +++ variants/kodedot/kodedot_partitions.csv | 7 ++ variants/kodedot/pins_arduino.h | 107 +++++++++++++++++++++++ 5 files changed, 201 insertions(+) create mode 100644 variants/kodedot/custom_ota_override.cpp create mode 100644 variants/kodedot/kodedot_partitions.csv create mode 100644 variants/kodedot/pins_arduino.h diff --git a/boards.txt b/boards.txt index b51a8840757..ceb0dc63d02 100644 --- a/boards.txt +++ b/boards.txt @@ -50547,3 +50547,60 @@ cyobot_v2_esp32s3.menu.ZigbeeMode.zczr.build.zigbee_mode=-DZIGBEE_MODE_ZCZR cyobot_v2_esp32s3.menu.ZigbeeMode.zczr.build.zigbee_libs=-lesp_zb_api.zczr -lzboss_stack.zczr -lzboss_port.remote ############################################################## + +kodedot.name=kode dot + +kodedot.bootloader.tool=esptool_py +kodedot.bootloader.tool.default=esptool_py + +kodedot.upload.tool=esptool_py_app_only +kodedot.upload.tool.default=esptool_py_app_only +kodedot.upload.tool.network=esp_ota + +kodedot.upload.maximum_size=8388608 +kodedot.upload.maximum_data_size=327680 +kodedot.upload.flags= +kodedot.upload.extra_flags= +kodedot.upload.use_1200bps_touch=false +kodedot.upload.wait_for_upload_port=false +kodedot.upload.speed=921600 + +kodedot.upload.erase_cmd= + +kodedot.serial.disableDTR=false +kodedot.serial.disableRTS=false + +kodedot.build.tarch=xtensa +kodedot.build.bootloader_addr=0x0 +kodedot.build.target=esp32s3 +kodedot.build.mcu=esp32s3 +kodedot.build.core=esp32 +kodedot.build.variant=kodedot +kodedot.build.board=KODE_DOT + +kodedot.build.usb_mode=1 +kodedot.build.cdc_on_boot=1 +kodedot.build.msc_on_boot=0 +kodedot.build.dfu_on_boot=0 + +kodedot.build.f_cpu=240000000L + +kodedot.build.flash_offset=0x400000 +kodedot.build.flash_size=16MB +kodedot.build.flash_freq=80m +kodedot.build.flash_mode=dio + +kodedot.build.custom_partitions=kodedot_partitions + +kodedot.build.psram_type=qspi +kodedot.build.defines= + +kodedot.build.loop_core=-DARDUINO_RUNNING_CORE=1 +kodedot.build.event_core=-DARDUINO_EVENT_RUNNING_CORE=1 + +kodedot.recipe.hooks.objcopy.postobjcopy.3.pattern= +kodedot.recipe.hooks.objcopy.postobjcopy.3.pattern_args= + +kodedot.recipe.output.save_file={build.project_name}.ino.bin + +############################################################## diff --git a/platform.txt b/platform.txt index 8c7f4432377..7bc89426323 100644 --- a/platform.txt +++ b/platform.txt @@ -330,3 +330,19 @@ tools.dfu-util.cmd=dfu-util tools.dfu-util.upload.params.verbose=-d tools.dfu-util.upload.params.quiet= tools.dfu-util.upload.pattern="{path}/{cmd}" --device {vid.0}:{pid.0} -D "{build.path}/{build.project_name}.bin" -Q + +## -------------------------------------------------------------------------- +## esptool_py_app_only is used to upload only the application image +## It won't upload the bootloader or any other binary except for the main application +## -------------------------------------------------------------------------- +tools.esptool_py_app_only.path={runtime.tools.esptool_py.path} +tools.esptool_py_app_only.cmd=esptool +tools.esptool_py_app_only.cmd.windows=esptool.exe + +tools.esptool_py_app_only.upload.protocol=serial +tools.esptool_py_app_only.upload.params.verbose= +tools.esptool_py_app_only.upload.params.quiet= + +tools.esptool_py_app_only.upload.pattern_args=--chip {build.mcu} --port "{serial.port}" --baud {upload.speed} {upload.flags} --before default_reset --after hard_reset write_flash --flash_mode {build.flash_mode} --flash_freq {build.flash_freq} --flash_size {build.flash_size} {build.flash_offset} "{build.path}/{build.project_name}.bin" {upload.extra_flags} + +tools.esptool_py_app_only.upload.pattern="{path}/{cmd}" {tools.esptool_py_app_only.upload.pattern_args} diff --git a/variants/kodedot/custom_ota_override.cpp b/variants/kodedot/custom_ota_override.cpp new file mode 100644 index 00000000000..a41db01db16 --- /dev/null +++ b/variants/kodedot/custom_ota_override.cpp @@ -0,0 +1,14 @@ +// custom_ota_override.cpp +// This function overrides the weak definition of `verifyRollbackLater()` in the kode dot board. + +extern "C" { +// Declare the weak function symbol to override it +bool verifyRollbackLater() __attribute__((weak)); +} + +// Custom implementation of verifyRollbackLater() +// Returning `true` prevents the OTA image from being automatically marked as valid. +// This ensures that the system will roll back to the previous image unless it is explicitly validated later. +bool verifyRollbackLater() { + return true; +} diff --git a/variants/kodedot/kodedot_partitions.csv b/variants/kodedot/kodedot_partitions.csv new file mode 100644 index 00000000000..f0732330850 --- /dev/null +++ b/variants/kodedot/kodedot_partitions.csv @@ -0,0 +1,7 @@ +# Name, Type, SubType, Offset, Size, Flags +nvs, data, nvs, 0x9000, 0x6000, +phy_init, data, phy, 0xf000, 0x1000, +otadata, data, ota, 0x10000, 0x2000, +ota_0, app, ota_0, 0x20000, 0x3E0000, +ota_1, app, ota_1, 0x400000, 0x800000, +storage, data, spiffs, 0xC00000, 0x400000, diff --git a/variants/kodedot/pins_arduino.h b/variants/kodedot/pins_arduino.h new file mode 100644 index 00000000000..f19ddb8a616 --- /dev/null +++ b/variants/kodedot/pins_arduino.h @@ -0,0 +1,107 @@ +/* + ──────────────────────────────────────────────────────────────────────── + KodeDot – ESP32-S3R8 Variant + Pin definition file for the Arduino-ESP32 core + ──────────────────────────────────────────────────────────────────────── + * External 2 × 10 connector → simple aliases PIN1 … PIN20 + * On-board QSPI LCD 410×502 @40 MHz (SPI3_HOST) + * micro-SD on SPI2_HOST + * Dual-I²C: external (GPIO37/36) + internal-sensors (GPIO48/47) + * USB VID/PID 0x303A:0x1001 +*/ + +#ifndef Pins_Arduino_h +#define Pins_Arduino_h + +#include +#include + +/*──────────────── USB device descriptor ────────────────*/ +#define USB_VID 0x303A // Espressif Systems VID +#define USB_PID 0x1001 // Product ID: KodeDot-S3 + +/*──────────────── UART0 (Arduino Serial) ────────────────*/ +static const uint8_t TX = 43; // U0TXD – PIN16 on the 2×10 header +static const uint8_t RX = 44; // U0RXD – PIN18 on the 2×10 header + +/*──────────────── I²C buses ─────────────────────────────*/ +/* External expansion bus → header pins 11/13 */ +static const uint8_t SCL = 37; // GPIO37 – PIN12 +static const uint8_t SDA = 36; // GPIO36 – PIN14 + +/* Internal sensor/touch bus (not on header) */ +#define INT_I2C_SCL 47 // GPIO47 +#define INT_I2C_SDA 48 // GPIO48 + +/*──────────────── SPI2 – micro-SD ───────────────────────*/ +static const uint8_t SS = 15; // SD_CS +static const uint8_t MOSI = 16; // SD_MOSI +static const uint8_t MISO = 18; // SD_MISO +static const uint8_t SCK = 17; // SD_CLK +#define BOARD_HAS_SD_SPI +#define SD_CS SS + +/*──────────────── QSPI LCD (SPI3_HOST) ─────────────────– + * Controller: ST7789 / 4-line SPI (no D/C pin) + * Resolution: 410×502 px, 16 bpp, RGB color-space + * Clock: 40 MHz + */ +#define BOARD_HAS_SPI_LCD +#define LCD_MODEL ST7789 +#define LCD_WIDTH 410 +#define LCD_HEIGHT 502 + +#define LCD_HOST SPI3_HOST +#define LCD_SCK 35 // GPIO35 • QSPI_CLK +#define LCD_MOSI 33 // GPIO33 • QSPI_IO0 (D0) +#define LCD_IO1 34 // GPIO34 • QSPI_IO1 (D1) +#define LCD_IO2 37 // GPIO37 • QSPI_IO2 (D2) +#define LCD_IO3 36 // GPIO36 • QSPI_IO3 (D3) +#define LCD_CS 10 // GPIO10 +#define LCD_RST 9 // GPIO09 +#define LCD_DC -1 // not used in 4-line SPI +/* Optional: back-light enable shares the NeoPixel pin */ +#define LCD_BL 5 // GPIO05 (same as NEOPIXEL) + +/*──────────────── Analog / Touch pads ────────────────*/ +static const uint8_t A0 = 11; // PIN4 – GPIO11 / TOUCH11 / ADC2_CH0 +static const uint8_t A1 = 12; // PIN6 – GPIO12 / TOUCH12 / ADC2_CH1 +static const uint8_t A2 = 13; // PIN8 – GPIO13 / TOUCH13 / ADC2_CH2 +static const uint8_t A3 = 14; // PIN10 – GPIO14 / TOUCH14 / ADC2_CH3 +static const uint8_t T0 = A0, T1 = A1, T2 = A2, T3 = A3; + +/*──────────────── On-board controls & indicator ─────────*/ +#define BUTTON_TOP 0 // GPIO00 – BOOT • active-LOW +#define BUTTON_BOTTOM 6 // GPIO06 • active-LOW +#define NEOPIXEL_PIN 5 // GPIO05 – WS2812 +#define LED_BUILTIN NEOPIXEL_PIN + +/*──────────────── JTAG (also on connector) ──────────────*/ +#define MTCK 39 // PIN11 – GPIO39 +#define MTDO 40 // PIN13 – GPIO40 +#define MTDI 41 // PIN15 – GPIO41 +#define MTMS 42 // PIN17 – GPIO42 + +/*──────────────── 2×10 header: simple aliases ─────────── + NOTE: power pins (1 = 5 V, 2 = 3 V3, 19/20 = GND) are **not** + exposed as GPIO numbers – they remain undefined here. */ +#define PIN3 1 // GPIO01 / TOUCH1 / ADC1_CH0 +#define PIN4 11 // GPIO11 / TOUCH11 / ADC2_CH0 +#define PIN5 2 // GPIO02 / TOUCH2 / ADC1_CH1 +#define PIN6 12 // GPIO12 / TOUCH12 / ADC2_CH1 +#define PIN7 3 // GPIO03 / TOUCH3 / ADC1_CH2 +#define PIN8 13 // GPIO13 / TOUCH13 / ADC2_CH2 +#define PIN9 4 // GPIO04 / TOUCH4 / ADC1_CH3 +#define PIN10 14 // GPIO14 / TOUCH14 / ADC2_CH3 +#define PIN11 39 // MTCK +#define PIN12 37 // SCL (external I²C) +#define PIN13 40 // MTDO +#define PIN14 36 // SDA (external I²C) +#define PIN15 41 // MTDI +#define PIN16 43 // TX (U0TXD) +#define PIN17 42 // MTMS +#define PIN18 44 // RX (U0RXD) +/* PIN1, PIN2, PIN19, PIN20 are power/ground and deliberately + left undefined – they are **not** usable as GPIO. */ + +#endif /* Pins_Arduino_h */ From 2592a7b3bb475554385b501a6a6f4a0d1b2fe060 Mon Sep 17 00:00:00 2001 From: Me No Dev Date: Mon, 30 Jun 2025 16:03:38 +0300 Subject: [PATCH 080/173] feat(p4): Add method to set the pins for SDIO to WiFi chip (#11513) * feat(p4): Add method to set the pins for SDIO to WiFi chip * ci(pre-commit): Apply automatic fixes --------- Co-authored-by: pre-commit-ci-lite[bot] <117423508+pre-commit-ci-lite[bot]@users.noreply.github.com> --- libraries/WiFi/src/WiFiGeneric.cpp | 86 +++++++++++++++++++++--------- libraries/WiFi/src/WiFiGeneric.h | 5 ++ 2 files changed, 67 insertions(+), 24 deletions(-) diff --git a/libraries/WiFi/src/WiFiGeneric.cpp b/libraries/WiFi/src/WiFiGeneric.cpp index bddcb5fd448..ec661ef7d8f 100644 --- a/libraries/WiFi/src/WiFiGeneric.cpp +++ b/libraries/WiFi/src/WiFiGeneric.cpp @@ -246,29 +246,67 @@ extern "C" { extern esp_err_t esp_hosted_init(); extern esp_err_t esp_hosted_deinit(); }; +typedef struct { + uint8_t pin_clk; + uint8_t pin_cmd; + uint8_t pin_d0; + uint8_t pin_d1; + uint8_t pin_d2; + uint8_t pin_d3; + uint8_t pin_reset; +} sdio_pin_config_t; + static bool hosted_initialized = false; +static sdio_pin_config_t sdio_pin_config = { +#ifdef BOARD_HAS_SDIO_ESP_HOSTED + .pin_clk = BOARD_SDIO_ESP_HOSTED_CLK, + .pin_cmd = BOARD_SDIO_ESP_HOSTED_CMD, + .pin_d0 = BOARD_SDIO_ESP_HOSTED_D0, + .pin_d1 = BOARD_SDIO_ESP_HOSTED_D1, + .pin_d2 = BOARD_SDIO_ESP_HOSTED_D2, + .pin_d3 = BOARD_SDIO_ESP_HOSTED_D3, + .pin_reset = BOARD_SDIO_ESP_HOSTED_RESET +#else + .pin_clk = CONFIG_ESP_SDIO_PIN_CLK, + .pin_cmd = CONFIG_ESP_SDIO_PIN_CMD, + .pin_d0 = CONFIG_ESP_SDIO_PIN_D0, + .pin_d1 = CONFIG_ESP_SDIO_PIN_D1, + .pin_d2 = CONFIG_ESP_SDIO_PIN_D2, + .pin_d3 = CONFIG_ESP_SDIO_PIN_D3, + .pin_reset = CONFIG_ESP_SDIO_GPIO_RESET_SLAVE +#endif +}; + +bool WiFiGenericClass::setPins(int8_t clk, int8_t cmd, int8_t d0, int8_t d1, int8_t d2, int8_t d3, int8_t rst) { + if (clk < 0 || cmd < 0 || d0 < 0 || d1 < 0 || d2 < 0 || d3 < 0 || rst < 0) { + log_e("All SDIO pins must be defined"); + return false; + } + if (hosted_initialized) { + log_e("SDIO pins must be set before WiFi is initialized"); + return false; + } + sdio_pin_config.pin_clk = clk; + sdio_pin_config.pin_cmd = cmd; + sdio_pin_config.pin_d0 = d0; + sdio_pin_config.pin_d1 = d1; + sdio_pin_config.pin_d2 = d2; + sdio_pin_config.pin_d3 = d3; + sdio_pin_config.pin_reset = rst; + return true; +} static bool wifiHostedInit() { if (!hosted_initialized) { hosted_initialized = true; struct esp_hosted_sdio_config conf = INIT_DEFAULT_HOST_SDIO_CONFIG(); -#ifdef BOARD_HAS_SDIO_ESP_HOSTED - conf.pin_clk.pin = BOARD_SDIO_ESP_HOSTED_CLK; - conf.pin_cmd.pin = BOARD_SDIO_ESP_HOSTED_CMD; - conf.pin_d0.pin = BOARD_SDIO_ESP_HOSTED_D0; - conf.pin_d1.pin = BOARD_SDIO_ESP_HOSTED_D1; - conf.pin_d2.pin = BOARD_SDIO_ESP_HOSTED_D2; - conf.pin_d3.pin = BOARD_SDIO_ESP_HOSTED_D3; - conf.pin_reset.pin = BOARD_SDIO_ESP_HOSTED_RESET; -#else - conf.pin_clk.pin = CONFIG_ESP_SDIO_PIN_CLK; - conf.pin_cmd.pin = CONFIG_ESP_SDIO_PIN_CMD; - conf.pin_d0.pin = CONFIG_ESP_SDIO_PIN_D0; - conf.pin_d1.pin = CONFIG_ESP_SDIO_PIN_D1; - conf.pin_d2.pin = CONFIG_ESP_SDIO_PIN_D2; - conf.pin_d3.pin = CONFIG_ESP_SDIO_PIN_D3; - conf.pin_reset.pin = CONFIG_ESP_SDIO_GPIO_RESET_SLAVE; -#endif + conf.pin_clk.pin = sdio_pin_config.pin_clk; + conf.pin_cmd.pin = sdio_pin_config.pin_cmd; + conf.pin_d0.pin = sdio_pin_config.pin_d0; + conf.pin_d1.pin = sdio_pin_config.pin_d1; + conf.pin_d2.pin = sdio_pin_config.pin_d2; + conf.pin_d3.pin = sdio_pin_config.pin_d3; + conf.pin_reset.pin = sdio_pin_config.pin_reset; // esp_hosted_sdio_set_config() will fail on second attempt but here temporarily to not cause exception on reinit if (esp_hosted_sdio_set_config(&conf) != ESP_OK || esp_hosted_init() != ESP_OK) { log_e("esp_hosted_init failed!"); @@ -279,13 +317,13 @@ static bool wifiHostedInit() { } // Attach pins to PeriMan here // Slave chip model is CONFIG_IDF_SLAVE_TARGET - // CONFIG_ESP_SDIO_PIN_CMD - // CONFIG_ESP_SDIO_PIN_CLK - // CONFIG_ESP_SDIO_PIN_D0 - // CONFIG_ESP_SDIO_PIN_D1 - // CONFIG_ESP_SDIO_PIN_D2 - // CONFIG_ESP_SDIO_PIN_D3 - // CONFIG_ESP_SDIO_GPIO_RESET_SLAVE + // sdio_pin_config.pin_clk + // sdio_pin_config.pin_cmd + // sdio_pin_config.pin_d0 + // sdio_pin_config.pin_d1 + // sdio_pin_config.pin_d2 + // sdio_pin_config.pin_d3 + // sdio_pin_config.pin_reset return true; } diff --git a/libraries/WiFi/src/WiFiGeneric.h b/libraries/WiFi/src/WiFiGeneric.h index ed216229ed4..bdfa7b5dd85 100644 --- a/libraries/WiFi/src/WiFiGeneric.h +++ b/libraries/WiFi/src/WiFiGeneric.h @@ -82,6 +82,11 @@ class WiFiGenericClass { public: WiFiGenericClass(); +#if CONFIG_ESP_WIFI_REMOTE_ENABLED + // Set SDIO pins for connection to external ESP MCU + static bool setPins(int8_t clk, int8_t cmd, int8_t d0, int8_t d1, int8_t d2, int8_t d3, int8_t rst); +#endif + wifi_event_id_t onEvent(WiFiEventCb cbEvent, arduino_event_id_t event = ARDUINO_EVENT_MAX); wifi_event_id_t onEvent(WiFiEventFuncCb cbEvent, arduino_event_id_t event = ARDUINO_EVENT_MAX); wifi_event_id_t onEvent(WiFiEventSysCb cbEvent, arduino_event_id_t event = ARDUINO_EVENT_MAX); From 8cf0818d824837ff1b13b63f10643c93adeecef6 Mon Sep 17 00:00:00 2001 From: Jason2866 <24528715+Jason2866@users.noreply.github.com> Date: Wed, 2 Jul 2025 13:17:00 +0200 Subject: [PATCH 081/173] make adresses for partitions.bin and boot_app0.bin configureable (#11534) --- tools/pioarduino-build.py | 10 ++++++++-- 1 file changed, 8 insertions(+), 2 deletions(-) diff --git a/tools/pioarduino-build.py b/tools/pioarduino-build.py index b6a0c0435c5..b67580e264c 100644 --- a/tools/pioarduino-build.py +++ b/tools/pioarduino-build.py @@ -216,8 +216,14 @@ def add_tinyuf2_extra_image(): "0x1000" if build_mcu in ["esp32", "esp32s2"] else ("0x2000" if build_mcu in ["esp32p4"] else "0x0000"), get_bootloader_image(variants_dir), ), - ("0x8000", join(env.subst("$BUILD_DIR"), "partitions.bin")), - ("0xe000", join(FRAMEWORK_DIR, "tools", "partitions", "boot_app0.bin")), + ( + board_config.get("upload.arduino.partitions_bin", "0x8000"), + join(env.subst("$BUILD_DIR"), "partitions.bin"), + ), + ( + board_config.get("upload.arduino.boot_app0", "0xe000"), + join(FRAMEWORK_DIR, "tools", "partitions", "boot_app0.bin"), + ), ] + [(offset, join(FRAMEWORK_DIR, img)) for offset, img in board_config.get("upload.arduino.flash_extra_images", [])], ) From f4fdecc60c465384e465a4b1d2bd1eac8f67912e Mon Sep 17 00:00:00 2001 From: Me No Dev Date: Wed, 2 Jul 2025 14:41:06 +0300 Subject: [PATCH 082/173] fix(csrf): Fix SCRF vulnerability in OTA examples and libraries (#11530) * fix(csrf): Fix SCRF vulnerability in WebUpdate.ino * fix(csrf): Prevent CSRF on other OTA examples * fix(csrf): Require auth user and pass to be changed * ci(pre-commit): Apply automatic fixes --------- Co-authored-by: pre-commit-ci-lite[bot] <117423508+pre-commit-ci-lite[bot]@users.noreply.github.com> --- .../HTTPUpdateServer/src/HTTPUpdateServer.h | 19 ++++++++ .../examples/OTAWebUpdater/OTAWebUpdater.ino | 35 ++++++++++++++- .../examples/WebUpdate/WebUpdate.ino | 43 ++++++++++++++++--- 3 files changed, 90 insertions(+), 7 deletions(-) diff --git a/libraries/HTTPUpdateServer/src/HTTPUpdateServer.h b/libraries/HTTPUpdateServer/src/HTTPUpdateServer.h index bb32bc03fdb..65d8cbaa783 100644 --- a/libraries/HTTPUpdateServer/src/HTTPUpdateServer.h +++ b/libraries/HTTPUpdateServer/src/HTTPUpdateServer.h @@ -27,6 +27,7 @@ static const char serverIndex[] PROGMEM = )"; static const char successResponse[] PROGMEM = "Update Success! Rebooting..."; +static const char *csrfHeaders[2] = {"Origin", "Host"}; class HTTPUpdateServer { public: @@ -56,6 +57,9 @@ class HTTPUpdateServer { _username = username; _password = password; + // collect headers for CSRF verification + _server->collectHeaders(csrfHeaders, 2); + // handler for the /update form page _server->on(path.c_str(), HTTP_GET, [&]() { if (_username != emptyString && _password != emptyString && !_server->authenticate(_username.c_str(), _password.c_str())) { @@ -69,6 +73,10 @@ class HTTPUpdateServer { path.c_str(), HTTP_POST, [&]() { if (!_authenticated) { + if (_username == emptyString || _password == emptyString) { + _server->send(200, F("text/html"), String(F("Update error: Wrong origin received!"))); + return; + } return _server->requestAuthentication(); } if (Update.hasError()) { @@ -100,6 +108,17 @@ class HTTPUpdateServer { return; } + String origin = _server->header(String(csrfHeaders[0])); + String host = _server->header(String(csrfHeaders[1])); + String expectedOrigin = String("http://") + host; + if (origin != expectedOrigin) { + if (_serial_output) { + Serial.printf("Wrong origin received! Expected: %s, Received: %s\n", expectedOrigin.c_str(), origin.c_str()); + } + _authenticated = false; + return; + } + if (_serial_output) { Serial.printf("Update: %s\n", upload.filename.c_str()); } diff --git a/libraries/Update/examples/OTAWebUpdater/OTAWebUpdater.ino b/libraries/Update/examples/OTAWebUpdater/OTAWebUpdater.ino index 7059bef4496..39d6cbce4af 100644 --- a/libraries/Update/examples/OTAWebUpdater/OTAWebUpdater.ino +++ b/libraries/Update/examples/OTAWebUpdater/OTAWebUpdater.ino @@ -8,10 +8,17 @@ #define SSID_FORMAT "ESP32-%06lX" // 12 chars total //#define PASSWORD "test123456" // generate if remarked +// Set the username and password for firmware upload +const char *authUser = "........"; +const char *authPass = "........"; + WebServer server(80); Ticker tkSecond; uint8_t otaDone = 0; +const char *csrfHeaders[2] = {"Origin", "Host"}; +static bool authenticated = false; + const char *alphanum = "0123456789!@#$%^&*abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ"; String generatePass(uint8_t str_len) { String buff; @@ -38,6 +45,9 @@ void apMode() { } void handleUpdateEnd() { + if (!authenticated) { + return server.requestAuthentication(); + } server.sendHeader("Connection", "close"); if (Update.hasError()) { server.send(502, "text/plain", Update.errorString()); @@ -45,6 +55,7 @@ void handleUpdateEnd() { server.sendHeader("Refresh", "10"); server.sendHeader("Location", "/"); server.send(307); + delay(500); ESP.restart(); } } @@ -56,18 +67,34 @@ void handleUpdate() { } HTTPUpload &upload = server.upload(); if (upload.status == UPLOAD_FILE_START) { + authenticated = server.authenticate(authUser, authPass); + if (!authenticated) { + Serial.println("Authentication fail!"); + otaDone = 0; + return; + } + String origin = server.header(String(csrfHeaders[0])); + String host = server.header(String(csrfHeaders[1])); + String expectedOrigin = String("http://") + host; + if (origin != expectedOrigin) { + Serial.printf("Wrong origin received! Expected: %s, Received: %s\n", expectedOrigin.c_str(), origin.c_str()); + authenticated = false; + otaDone = 0; + return; + } + Serial.printf("Receiving Update: %s, Size: %d\n", upload.filename.c_str(), fsize); if (!Update.begin(fsize)) { otaDone = 0; Update.printError(Serial); } - } else if (upload.status == UPLOAD_FILE_WRITE) { + } else if (authenticated && upload.status == UPLOAD_FILE_WRITE) { if (Update.write(upload.buf, upload.currentSize) != upload.currentSize) { Update.printError(Serial); } else { otaDone = 100 * Update.progress() / Update.size(); } - } else if (upload.status == UPLOAD_FILE_END) { + } else if (authenticated && upload.status == UPLOAD_FILE_END) { if (Update.end(true)) { Serial.printf("Update Success: %u bytes\nRebooting...\n", upload.totalSize); } else { @@ -78,6 +105,7 @@ void handleUpdate() { } void webServerInit() { + server.collectHeaders(csrfHeaders, 2); server.on( "/update", HTTP_POST, []() { @@ -92,6 +120,9 @@ void webServerInit() { server.send_P(200, "image/x-icon", favicon_ico_gz, favicon_ico_gz_len); }); server.onNotFound([]() { + if (!server.authenticate(authUser, authPass)) { + return server.requestAuthentication(); + } server.send(200, "text/html", indexHtml); }); server.begin(); diff --git a/libraries/WebServer/examples/WebUpdate/WebUpdate.ino b/libraries/WebServer/examples/WebUpdate/WebUpdate.ino index 10ddb5e7b64..9e45de7d985 100644 --- a/libraries/WebServer/examples/WebUpdate/WebUpdate.ino +++ b/libraries/WebServer/examples/WebUpdate/WebUpdate.ino @@ -12,10 +12,17 @@ const char *host = "esp32-webupdate"; const char *ssid = "........"; const char *password = "........"; +// Set the username and password for firmware upload +const char *authUser = "........"; +const char *authPass = "........"; + WebServer server(80); const char *serverIndex = "
"; +const char *csrfHeaders[2] = {"Origin", "Host"}; +static bool authenticated = false; + void setup(void) { Serial.begin(115200); Serial.println(); @@ -24,37 +31,63 @@ void setup(void) { WiFi.begin(ssid, password); if (WiFi.waitForConnectResult() == WL_CONNECTED) { MDNS.begin(host); + server.collectHeaders(csrfHeaders, 2); server.on("/", HTTP_GET, []() { + if (!server.authenticate(authUser, authPass)) { + return server.requestAuthentication(); + } server.sendHeader("Connection", "close"); server.send(200, "text/html", serverIndex); }); server.on( "/update", HTTP_POST, []() { + if (!authenticated) { + return server.requestAuthentication(); + } server.sendHeader("Connection", "close"); - server.send(200, "text/plain", (Update.hasError()) ? "FAIL" : "OK"); - ESP.restart(); + if (Update.hasError()) { + server.send(200, "text/plain", "FAIL"); + } else { + server.send(200, "text/plain", "Success! Rebooting..."); + delay(500); + ESP.restart(); + } }, []() { HTTPUpload &upload = server.upload(); if (upload.status == UPLOAD_FILE_START) { Serial.setDebugOutput(true); + authenticated = server.authenticate(authUser, authPass); + if (!authenticated) { + Serial.println("Authentication fail!"); + return; + } + String origin = server.header(String(csrfHeaders[0])); + String host = server.header(String(csrfHeaders[1])); + String expectedOrigin = String("http://") + host; + if (origin != expectedOrigin) { + Serial.printf("Wrong origin received! Expected: %s, Received: %s\n", expectedOrigin.c_str(), origin.c_str()); + authenticated = false; + return; + } + Serial.printf("Update: %s\n", upload.filename.c_str()); if (!Update.begin()) { //start with max available size Update.printError(Serial); } - } else if (upload.status == UPLOAD_FILE_WRITE) { + } else if (authenticated && upload.status == UPLOAD_FILE_WRITE) { if (Update.write(upload.buf, upload.currentSize) != upload.currentSize) { Update.printError(Serial); } - } else if (upload.status == UPLOAD_FILE_END) { + } else if (authenticated && upload.status == UPLOAD_FILE_END) { if (Update.end(true)) { //true to set the size to the current progress Serial.printf("Update Success: %u\nRebooting...\n", upload.totalSize); } else { Update.printError(Serial); } Serial.setDebugOutput(false); - } else { + } else if (authenticated) { Serial.printf("Update Failed Unexpectedly (likely broken connection): status=%d\n", upload.status); } } From 212b12b625f7dca9b646a2ae5c422936c7ef0ab6 Mon Sep 17 00:00:00 2001 From: Me No Dev Date: Wed, 2 Jul 2025 14:41:39 +0300 Subject: [PATCH 083/173] IDF release/v5.4 (#11517) * IDF release/v5.4 dfa785ed * IDF release/v5.4 72775cd6 * IDF release/v5.4 858a988d --- package/package_esp32_index.template.json | 68 +++++++++++------------ 1 file changed, 34 insertions(+), 34 deletions(-) diff --git a/package/package_esp32_index.template.json b/package/package_esp32_index.template.json index 9c1516dbacd..bbb77f8ef5a 100644 --- a/package/package_esp32_index.template.json +++ b/package/package_esp32_index.template.json @@ -51,7 +51,7 @@ { "packager": "esp32", "name": "esp32-arduino-libs", - "version": "idf-release_v5.4-f0f2980d-v1" + "version": "idf-release_v5.4-858a988d-v1" }, { "packager": "esp32", @@ -104,63 +104,63 @@ "tools": [ { "name": "esp32-arduino-libs", - "version": "idf-release_v5.4-f0f2980d-v1", + "version": "idf-release_v5.4-858a988d-v1", "systems": [ { "host": "i686-mingw32", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.4/esp32-arduino-libs-idf-release_v5.4-f0f2980d-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.4-f0f2980d-v1.zip", - "checksum": "SHA-256:b1a77c83634111d78a356f1d1756dc815eeeb91605c5d8714a0db7cccbd0bede", - "size": "353978504" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.4/esp32-arduino-libs-idf-release_v5.4-858a988d-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.4-858a988d-v1.zip", + "checksum": "SHA-256:4fc6eace642735036404c722d5602a379fe16378c72ffd060096cbef1c62d69d", + "size": "354134726" }, { "host": "x86_64-mingw32", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.4/esp32-arduino-libs-idf-release_v5.4-f0f2980d-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.4-f0f2980d-v1.zip", - "checksum": "SHA-256:b1a77c83634111d78a356f1d1756dc815eeeb91605c5d8714a0db7cccbd0bede", - "size": "353978504" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.4/esp32-arduino-libs-idf-release_v5.4-858a988d-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.4-858a988d-v1.zip", + "checksum": "SHA-256:4fc6eace642735036404c722d5602a379fe16378c72ffd060096cbef1c62d69d", + "size": "354134726" }, { "host": "arm64-apple-darwin", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.4/esp32-arduino-libs-idf-release_v5.4-f0f2980d-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.4-f0f2980d-v1.zip", - "checksum": "SHA-256:b1a77c83634111d78a356f1d1756dc815eeeb91605c5d8714a0db7cccbd0bede", - "size": "353978504" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.4/esp32-arduino-libs-idf-release_v5.4-858a988d-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.4-858a988d-v1.zip", + "checksum": "SHA-256:4fc6eace642735036404c722d5602a379fe16378c72ffd060096cbef1c62d69d", + "size": "354134726" }, { "host": "x86_64-apple-darwin", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.4/esp32-arduino-libs-idf-release_v5.4-f0f2980d-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.4-f0f2980d-v1.zip", - "checksum": "SHA-256:b1a77c83634111d78a356f1d1756dc815eeeb91605c5d8714a0db7cccbd0bede", - "size": "353978504" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.4/esp32-arduino-libs-idf-release_v5.4-858a988d-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.4-858a988d-v1.zip", + "checksum": "SHA-256:4fc6eace642735036404c722d5602a379fe16378c72ffd060096cbef1c62d69d", + "size": "354134726" }, { "host": "x86_64-pc-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.4/esp32-arduino-libs-idf-release_v5.4-f0f2980d-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.4-f0f2980d-v1.zip", - "checksum": "SHA-256:b1a77c83634111d78a356f1d1756dc815eeeb91605c5d8714a0db7cccbd0bede", - "size": "353978504" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.4/esp32-arduino-libs-idf-release_v5.4-858a988d-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.4-858a988d-v1.zip", + "checksum": "SHA-256:4fc6eace642735036404c722d5602a379fe16378c72ffd060096cbef1c62d69d", + "size": "354134726" }, { "host": "i686-pc-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.4/esp32-arduino-libs-idf-release_v5.4-f0f2980d-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.4-f0f2980d-v1.zip", - "checksum": "SHA-256:b1a77c83634111d78a356f1d1756dc815eeeb91605c5d8714a0db7cccbd0bede", - "size": "353978504" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.4/esp32-arduino-libs-idf-release_v5.4-858a988d-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.4-858a988d-v1.zip", + "checksum": "SHA-256:4fc6eace642735036404c722d5602a379fe16378c72ffd060096cbef1c62d69d", + "size": "354134726" }, { "host": "aarch64-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.4/esp32-arduino-libs-idf-release_v5.4-f0f2980d-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.4-f0f2980d-v1.zip", - "checksum": "SHA-256:b1a77c83634111d78a356f1d1756dc815eeeb91605c5d8714a0db7cccbd0bede", - "size": "353978504" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.4/esp32-arduino-libs-idf-release_v5.4-858a988d-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.4-858a988d-v1.zip", + "checksum": "SHA-256:4fc6eace642735036404c722d5602a379fe16378c72ffd060096cbef1c62d69d", + "size": "354134726" }, { "host": "arm-linux-gnueabihf", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.4/esp32-arduino-libs-idf-release_v5.4-f0f2980d-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.4-f0f2980d-v1.zip", - "checksum": "SHA-256:b1a77c83634111d78a356f1d1756dc815eeeb91605c5d8714a0db7cccbd0bede", - "size": "353978504" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.4/esp32-arduino-libs-idf-release_v5.4-858a988d-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.4-858a988d-v1.zip", + "checksum": "SHA-256:4fc6eace642735036404c722d5602a379fe16378c72ffd060096cbef1c62d69d", + "size": "354134726" } ] }, From bc7a549876c33686f6d9050e71a2f045717c41e2 Mon Sep 17 00:00:00 2001 From: HighDoping Date: Wed, 2 Jul 2025 19:54:12 +0800 Subject: [PATCH 084/173] fix(example): led flash not working if not using default model in camera example. (#11466) * fix(example): led flash not working if not using default model in camera example fix(example): add camera_config.h and enable LED FLASH based on board model fix(example): Remove face detection description as no longer supported * fix(example): add header guard for board_config.h --- .../CameraWebServer/CameraWebServer.ino | 39 +++---------------- .../Camera/CameraWebServer/app_httpd.cpp | 30 +++++++------- .../Camera/CameraWebServer/board_config.h | 34 ++++++++++++++++ 3 files changed, 53 insertions(+), 50 deletions(-) create mode 100644 libraries/ESP32/examples/Camera/CameraWebServer/board_config.h diff --git a/libraries/ESP32/examples/Camera/CameraWebServer/CameraWebServer.ino b/libraries/ESP32/examples/Camera/CameraWebServer/CameraWebServer.ino index d483e11b1df..83733d4b5cf 100644 --- a/libraries/ESP32/examples/Camera/CameraWebServer/CameraWebServer.ino +++ b/libraries/ESP32/examples/Camera/CameraWebServer/CameraWebServer.ino @@ -1,37 +1,10 @@ #include "esp_camera.h" #include -// -// WARNING!!! PSRAM IC required for UXGA resolution and high JPEG quality -// Ensure ESP32 Wrover Module or other board with PSRAM is selected -// Partial images will be transmitted if image exceeds buffer size -// -// You must select partition scheme from the board menu that has at least 3MB APP space. -// Face Recognition is DISABLED for ESP32 and ESP32-S2, because it takes up from 15 -// seconds to process single frame. Face Detection is ENABLED if PSRAM is enabled as well - -// =================== -// Select camera model -// =================== -//#define CAMERA_MODEL_WROVER_KIT // Has PSRAM -#define CAMERA_MODEL_ESP_EYE // Has PSRAM -//#define CAMERA_MODEL_ESP32S3_EYE // Has PSRAM -//#define CAMERA_MODEL_M5STACK_PSRAM // Has PSRAM -//#define CAMERA_MODEL_M5STACK_V2_PSRAM // M5Camera version B Has PSRAM -//#define CAMERA_MODEL_M5STACK_WIDE // Has PSRAM -//#define CAMERA_MODEL_M5STACK_ESP32CAM // No PSRAM -//#define CAMERA_MODEL_M5STACK_UNITCAM // No PSRAM -//#define CAMERA_MODEL_M5STACK_CAMS3_UNIT // Has PSRAM -//#define CAMERA_MODEL_AI_THINKER // Has PSRAM -//#define CAMERA_MODEL_TTGO_T_JOURNAL // No PSRAM -//#define CAMERA_MODEL_XIAO_ESP32S3 // Has PSRAM -// ** Espressif Internal Boards ** -//#define CAMERA_MODEL_ESP32_CAM_BOARD -//#define CAMERA_MODEL_ESP32S2_CAM_BOARD -//#define CAMERA_MODEL_ESP32S3_CAM_LCD -//#define CAMERA_MODEL_DFRobot_FireBeetle2_ESP32S3 // Has PSRAM -//#define CAMERA_MODEL_DFRobot_Romeo_ESP32S3 // Has PSRAM -#include "camera_pins.h" +// =========================== +// Select camera model in board_config.h +// =========================== +#include "board_config.h" // =========================== // Enter your WiFi credentials @@ -40,7 +13,7 @@ const char *ssid = "**********"; const char *password = "**********"; void startCameraServer(); -void setupLedFlash(int pin); +void setupLedFlash(); void setup() { Serial.begin(115200); @@ -130,7 +103,7 @@ void setup() { // Setup LED FLash if LED pin is defined in camera_pins.h #if defined(LED_GPIO_NUM) - setupLedFlash(LED_GPIO_NUM); + setupLedFlash(); #endif WiFi.begin(ssid, password); diff --git a/libraries/ESP32/examples/Camera/CameraWebServer/app_httpd.cpp b/libraries/ESP32/examples/Camera/CameraWebServer/app_httpd.cpp index cc924bd5b3b..589fea33b7f 100644 --- a/libraries/ESP32/examples/Camera/CameraWebServer/app_httpd.cpp +++ b/libraries/ESP32/examples/Camera/CameraWebServer/app_httpd.cpp @@ -19,18 +19,14 @@ #include "esp32-hal-ledc.h" #include "sdkconfig.h" #include "camera_index.h" +#include "board_config.h" #if defined(ARDUINO_ARCH_ESP32) && defined(CONFIG_ARDUHAL_ESP_LOG) #include "esp32-hal-log.h" #endif -// Enable LED FLASH setting -#define CONFIG_LED_ILLUMINATOR_ENABLED 1 - // LED FLASH setup -#if CONFIG_LED_ILLUMINATOR_ENABLED - -#define LED_LEDC_GPIO 22 //configure LED pin +#if defined(LED_GPIO_NUM) #define CONFIG_LED_MAX_INTENSITY 255 int led_duty = 0; @@ -91,13 +87,13 @@ static int ra_filter_run(ra_filter_t *filter, int value) { } #endif -#if CONFIG_LED_ILLUMINATOR_ENABLED +#if defined(LED_GPIO_NUM) void enable_led(bool en) { // Turn LED On or Off int duty = en ? led_duty : 0; if (en && isStreaming && (led_duty > CONFIG_LED_MAX_INTENSITY)) { duty = CONFIG_LED_MAX_INTENSITY; } - ledcWrite(LED_LEDC_GPIO, duty); + ledcWrite(LED_GPIO_NUM, duty); //ledc_set_duty(CONFIG_LED_LEDC_SPEED_MODE, CONFIG_LED_LEDC_CHANNEL, duty); //ledc_update_duty(CONFIG_LED_LEDC_SPEED_MODE, CONFIG_LED_LEDC_CHANNEL); log_i("Set LED intensity to %d", duty); @@ -162,7 +158,7 @@ static esp_err_t capture_handler(httpd_req_t *req) { int64_t fr_start = esp_timer_get_time(); #endif -#if CONFIG_LED_ILLUMINATOR_ENABLED +#if defined(LED_GPIO_NUM) enable_led(true); vTaskDelay(150 / portTICK_PERIOD_MS); // The LED needs to be turned on ~150ms before the call to esp_camera_fb_get() fb = esp_camera_fb_get(); // or it won't be visible in the frame. A better way to do this is needed. @@ -230,7 +226,7 @@ static esp_err_t stream_handler(httpd_req_t *req) { httpd_resp_set_hdr(req, "Access-Control-Allow-Origin", "*"); httpd_resp_set_hdr(req, "X-Framerate", "60"); -#if CONFIG_LED_ILLUMINATOR_ENABLED +#if defined(LED_GPIO_NUM) isStreaming = true; enable_led(true); #endif @@ -293,7 +289,7 @@ static esp_err_t stream_handler(httpd_req_t *req) { ); } -#if CONFIG_LED_ILLUMINATOR_ENABLED +#if defined(LED_GPIO_NUM) isStreaming = false; enable_led(false); #endif @@ -393,7 +389,7 @@ static esp_err_t cmd_handler(httpd_req_t *req) { } else if (!strcmp(variable, "ae_level")) { res = s->set_ae_level(s, val); } -#if CONFIG_LED_ILLUMINATOR_ENABLED +#if defined(LED_GPIO_NUM) else if (!strcmp(variable, "led_intensity")) { led_duty = val; if (isStreaming) { @@ -481,7 +477,7 @@ static esp_err_t status_handler(httpd_req_t *req) { p += sprintf(p, "\"vflip\":%u,", s->status.vflip); p += sprintf(p, "\"dcw\":%u,", s->status.dcw); p += sprintf(p, "\"colorbar\":%u", s->status.colorbar); -#if CONFIG_LED_ILLUMINATOR_ENABLED +#if defined(LED_GPIO_NUM) p += sprintf(p, ",\"led_intensity\":%u", led_duty); #else p += sprintf(p, ",\"led_intensity\":%d", -1); @@ -843,10 +839,10 @@ void startCameraServer() { } } -void setupLedFlash(int pin) { -#if CONFIG_LED_ILLUMINATOR_ENABLED - ledcAttach(pin, 5000, 8); +void setupLedFlash() { +#if defined(LED_GPIO_NUM) + ledcAttach(LED_GPIO_NUM, 5000, 8); #else - log_i("LED flash is disabled -> CONFIG_LED_ILLUMINATOR_ENABLED = 0"); + log_i("LED flash is disabled -> LED_GPIO_NUM undefined"); #endif } diff --git a/libraries/ESP32/examples/Camera/CameraWebServer/board_config.h b/libraries/ESP32/examples/Camera/CameraWebServer/board_config.h new file mode 100644 index 00000000000..ca7edbab73e --- /dev/null +++ b/libraries/ESP32/examples/Camera/CameraWebServer/board_config.h @@ -0,0 +1,34 @@ +#ifndef BOARD_CONFIG_H +#define BOARD_CONFIG_H + +// +// WARNING!!! PSRAM IC required for UXGA resolution and high JPEG quality +// Ensure ESP32 Wrover Module or other board with PSRAM is selected +// Partial images will be transmitted if image exceeds buffer size +// +// You must select partition scheme from the board menu that has at least 3MB APP space. + +// =================== +// Select camera model +// =================== +//#define CAMERA_MODEL_WROVER_KIT // Has PSRAM +#define CAMERA_MODEL_ESP_EYE // Has PSRAM +//#define CAMERA_MODEL_ESP32S3_EYE // Has PSRAM +//#define CAMERA_MODEL_M5STACK_PSRAM // Has PSRAM +//#define CAMERA_MODEL_M5STACK_V2_PSRAM // M5Camera version B Has PSRAM +//#define CAMERA_MODEL_M5STACK_WIDE // Has PSRAM +//#define CAMERA_MODEL_M5STACK_ESP32CAM // No PSRAM +//#define CAMERA_MODEL_M5STACK_UNITCAM // No PSRAM +//#define CAMERA_MODEL_M5STACK_CAMS3_UNIT // Has PSRAM +//#define CAMERA_MODEL_AI_THINKER // Has PSRAM +//#define CAMERA_MODEL_TTGO_T_JOURNAL // No PSRAM +//#define CAMERA_MODEL_XIAO_ESP32S3 // Has PSRAM +// ** Espressif Internal Boards ** +//#define CAMERA_MODEL_ESP32_CAM_BOARD +//#define CAMERA_MODEL_ESP32S2_CAM_BOARD +//#define CAMERA_MODEL_ESP32S3_CAM_LCD +//#define CAMERA_MODEL_DFRobot_FireBeetle2_ESP32S3 // Has PSRAM +//#define CAMERA_MODEL_DFRobot_Romeo_ESP32S3 // Has PSRAM +#include "camera_pins.h" + +#endif // BOARD_CONFIG_H From 7f75e445f7e3ff20f44da48a15160b247d93510e Mon Sep 17 00:00:00 2001 From: "Daniel.Cao" <144674500+DanielCao0@users.noreply.github.com> Date: Wed, 2 Jul 2025 20:23:01 +0800 Subject: [PATCH 085/173] feat(board): add support for RAKwireless RAK3112 (#11485) * feat(rak3112): add pins_arduino.h for RAKWireless RAK3112 module * feat(rak3112): update pins_arduino.h to define LED pins and update board.txt * Delete the redundant configuration information in board.txt * Restore the incorrect modifications to board.txt * Delete blank lines * Move the rak configuration information to the end of the boards.txt . * ci(pre-commit): Apply automatic fixes --------- Co-authored-by: Daniel.Cao Co-authored-by: pre-commit-ci-lite[bot] <117423508+pre-commit-ci-lite[bot]@users.noreply.github.com> Co-authored-by: Me No Dev --- boards.txt | 158 ++++++++++++++++++++ variants/rakwireless_rak3112/pins_arduino.h | 50 +++++++ 2 files changed, 208 insertions(+) create mode 100644 variants/rakwireless_rak3112/pins_arduino.h diff --git a/boards.txt b/boards.txt index ceb0dc63d02..fe67071ea17 100644 --- a/boards.txt +++ b/boards.txt @@ -50548,6 +50548,164 @@ cyobot_v2_esp32s3.menu.ZigbeeMode.zczr.build.zigbee_libs=-lesp_zb_api.zczr -lzbo ############################################################## +rakwireless_rak3112.name=RAKwireless RAK3112 + +rakwireless_rak3112.upload.tool=esptool_py +rakwireless_rak3112.upload.tool.default=esptool_py +rakwireless_rak3112.upload.tool.network=esp_ota +rakwireless_rak3112.upload.maximum_size=1310720 +rakwireless_rak3112.upload.maximum_data_size=327680 +rakwireless_rak3112.upload.wait_for_upload_port=false +rakwireless_rak3112.upload.speed=460800 +rakwireless_rak3112.upload.flags= +rakwireless_rak3112.upload.extra_flags= + +rakwireless_rak3112.bootloader.tool=esptool_py +rakwireless_rak3112.bootloader.tool.default=esptool_py + +rakwireless_rak3112.serial.disableDTR=true +rakwireless_rak3112.serial.disableRTS=true + +rakwireless_rak3112.build.tarch=xtensa +rakwireless_rak3112.build.bootloader_addr=0x0 +rakwireless_rak3112.build.mcu=esp32s3 +rakwireless_rak3112.build.core=esp32 +rakwireless_rak3112.build.target=esp32s3 +rakwireless_rak3112.build.variant=rakwireless_rak3112 +rakwireless_rak3112.build.board=RAKWIRELESS_RAK3112 + +rakwireless_rak3112.build.usb_mode=1 +rakwireless_rak3112.build.cdc_on_boot=1 +rakwireless_rak3112.build.msc_on_boot=0 +rakwireless_rak3112.build.dfu_on_boot=0 + +rakwireless_rak3112.build.f_cpu=240000000L +rakwireless_rak3112.build.flash_size=16MB +rakwireless_rak3112.build.flash_freq=80m +rakwireless_rak3112.build.flash_mode=dio +rakwireless_rak3112.build.boot=dio +rakwireless_rak3112.build.partitions=default +rakwireless_rak3112.build.defines= + +rakwireless_rak3112.menu.PSRAM.enabled=Enabled +rakwireless_rak3112.menu.PSRAM.enabled.build.defines=-DBOARD_HAS_PSRAM +rakwireless_rak3112.menu.PSRAM.enabled.build.psram_type=opi +rakwireless_rak3112.menu.PSRAM.disabled=Disabled +rakwireless_rak3112.menu.PSRAM.disabled.build.defines= + +rakwireless_rak3112.menu.LoopCore.1=Core 1 +rakwireless_rak3112.menu.LoopCore.1.build.loop_core=-DARDUINO_RUNNING_CORE=1 +rakwireless_rak3112.menu.LoopCore.0=Core 0 +rakwireless_rak3112.menu.LoopCore.0.build.loop_core=-DARDUINO_RUNNING_CORE=0 + +rakwireless_rak3112.menu.EventsCore.1=Core 1 +rakwireless_rak3112.menu.EventsCore.1.build.event_core=-DARDUINO_EVENT_RUNNING_CORE=1 +rakwireless_rak3112.menu.EventsCore.0=Core 0 +rakwireless_rak3112.menu.EventsCore.0.build.event_core=-DARDUINO_EVENT_RUNNING_CORE=0 + +rakwireless_rak3112.menu.USBMode.hwcdc=Hardware CDC and JTAG +rakwireless_rak3112.menu.USBMode.hwcdc.build.usb_mode=1 +rakwireless_rak3112.menu.USBMode.default=USB-OTG (TinyUSB) +rakwireless_rak3112.menu.USBMode.default.build.usb_mode=0 + +rakwireless_rak3112.menu.CDCOnBoot.default=Enabled +rakwireless_rak3112.menu.CDCOnBoot.default.build.cdc_on_boot=1 +rakwireless_rak3112.menu.CDCOnBoot.cdc=Disabled +rakwireless_rak3112.menu.CDCOnBoot.cdc.build.cdc_on_boot=0 + +rakwireless_rak3112.menu.MSCOnBoot.default=Disabled +rakwireless_rak3112.menu.MSCOnBoot.default.build.msc_on_boot=0 +rakwireless_rak3112.menu.MSCOnBoot.msc=Enabled (Requires USB-OTG Mode) +rakwireless_rak3112.menu.MSCOnBoot.msc.build.msc_on_boot=1 + +rakwireless_rak3112.menu.DFUOnBoot.default=Disabled +rakwireless_rak3112.menu.DFUOnBoot.default.build.dfu_on_boot=0 +rakwireless_rak3112.menu.DFUOnBoot.dfu=Enabled (Requires USB-OTG Mode) +rakwireless_rak3112.menu.DFUOnBoot.dfu.build.dfu_on_boot=1 + +rakwireless_rak3112.menu.UploadMode.default=UART0 / Hardware CDC +rakwireless_rak3112.menu.UploadMode.default.upload.use_1200bps_touch=false +rakwireless_rak3112.menu.UploadMode.default.upload.wait_for_upload_port=false +rakwireless_rak3112.menu.UploadMode.cdc=USB-OTG CDC (TinyUSB) +rakwireless_rak3112.menu.UploadMode.cdc.upload.use_1200bps_touch=true +rakwireless_rak3112.menu.UploadMode.cdc.upload.wait_for_upload_port=true + +rakwireless_rak3112.menu.PartitionScheme.default_16MB=Default (6.25MB APP/3.43MB SPIFFS) +rakwireless_rak3112.menu.PartitionScheme.default_16MB.build.partitions=default_16MB +rakwireless_rak3112.menu.PartitionScheme.default_16MB.upload.maximum_size=6553600 +rakwireless_rak3112.menu.PartitionScheme.app3M_fat9M_16MB=16M Flash (3MB APP/9MB FATFS) +rakwireless_rak3112.menu.PartitionScheme.app3M_fat9M_16MB.build.partitions=app3M_fat9M_16MB +rakwireless_rak3112.menu.PartitionScheme.app3M_fat9M_16MB.upload.maximum_size=3145728 +rakwireless_rak3112.menu.PartitionScheme.tinyuf2=TinyUF2 16MB (2MB APP/11.6MB FATFS) +rakwireless_rak3112.menu.PartitionScheme.tinyuf2.build.custom_bootloader=bootloader-tinyuf2 +rakwireless_rak3112.menu.PartitionScheme.tinyuf2.build.partitions=tinyuf2-partitions-16MB +rakwireless_rak3112.menu.PartitionScheme.tinyuf2.upload.maximum_size=2097152 +rakwireless_rak3112.menu.PartitionScheme.tinyuf2.upload.extra_flags=0x410000 "{runtime.platform.path}/variants/{build.variant}/tinyuf2.bin" +rakwireless_rak3112.menu.PartitionScheme.tinyuf2_noota=TinyUF2 16MB No OTA(4MB APP/11.6MB FATFS) +rakwireless_rak3112.menu.PartitionScheme.tinyuf2_noota.build.custom_bootloader=bootloader-tinyuf2 +rakwireless_rak3112.menu.PartitionScheme.tinyuf2_noota.build.partitions=tinyuf2-partitions-16MB-noota +rakwireless_rak3112.menu.PartitionScheme.tinyuf2_noota.upload.maximum_size=4194304 +rakwireless_rak3112.menu.PartitionScheme.tinyuf2_noota.upload.extra_flags=0x410000 "{runtime.platform.path}/variants/{build.variant}/tinyuf2.bin" +rakwireless_rak3112.menu.PartitionScheme.large_spiffs=Large SPIFFS (4.5MB APP/6.93MB SPIFFS) +rakwireless_rak3112.menu.PartitionScheme.large_spiffs.build.partitions=large_spiffs_16MB +rakwireless_rak3112.menu.PartitionScheme.large_spiffs.upload.maximum_size=4718592 +rakwireless_rak3112.menu.PartitionScheme.custom=Custom +rakwireless_rak3112.menu.PartitionScheme.custom.build.partitions= +rakwireless_rak3112.menu.PartitionScheme.custom.upload.maximum_size=16777216 + +rakwireless_rak3112.menu.CPUFreq.240=240MHz (WiFi/BT) +rakwireless_rak3112.menu.CPUFreq.240.build.f_cpu=240000000L +rakwireless_rak3112.menu.CPUFreq.160=160MHz (WiFi/BT) +rakwireless_rak3112.menu.CPUFreq.160.build.f_cpu=160000000L +rakwireless_rak3112.menu.CPUFreq.80=80MHz (WiFi/BT) +rakwireless_rak3112.menu.CPUFreq.80.build.f_cpu=80000000L + +rakwireless_rak3112.menu.FlashMode.qio=QIO +rakwireless_rak3112.menu.FlashMode.qio.build.flash_mode=dio +rakwireless_rak3112.menu.FlashMode.qio.build.boot=qio +rakwireless_rak3112.menu.FlashMode.dio=DIO +rakwireless_rak3112.menu.FlashMode.dio.build.flash_mode=dio +rakwireless_rak3112.menu.FlashMode.dio.build.boot=dio + +rakwireless_rak3112.menu.FlashFreq.80=80MHz +rakwireless_rak3112.menu.FlashFreq.80.build.flash_freq=80m +rakwireless_rak3112.menu.FlashFreq.40=40MHz +rakwireless_rak3112.menu.FlashFreq.40.build.flash_freq=40m + +rakwireless_rak3112.menu.UploadSpeed.921600=921600 +rakwireless_rak3112.menu.UploadSpeed.921600.upload.speed=921600 +rakwireless_rak3112.menu.UploadSpeed.115200=115200 +rakwireless_rak3112.menu.UploadSpeed.115200.upload.speed=115200 +rakwireless_rak3112.menu.UploadSpeed.256000.windows=256000 +rakwireless_rak3112.menu.UploadSpeed.256000.upload.speed=256000 +rakwireless_rak3112.menu.UploadSpeed.230400.windows.upload.speed=256000 +rakwireless_rak3112.menu.UploadSpeed.230400=230400 +rakwireless_rak3112.menu.UploadSpeed.230400.upload.speed=230400 +rakwireless_rak3112.menu.UploadSpeed.460800.linux=460800 +rakwireless_rak3112.menu.UploadSpeed.460800.macosx=460800 +rakwireless_rak3112.menu.UploadSpeed.460800.upload.speed=460800 +rakwireless_rak3112.menu.UploadSpeed.512000.windows=512000 +rakwireless_rak3112.menu.UploadSpeed.512000.upload.speed=512000 + +rakwireless_rak3112.menu.DebugLevel.none=None +rakwireless_rak3112.menu.DebugLevel.none.build.code_debug=0 +rakwireless_rak3112.menu.DebugLevel.error=Error +rakwireless_rak3112.menu.DebugLevel.error.build.code_debug=1 +rakwireless_rak3112.menu.DebugLevel.warn=Warn +rakwireless_rak3112.menu.DebugLevel.warn.build.code_debug=2 +rakwireless_rak3112.menu.DebugLevel.info=Info +rakwireless_rak3112.menu.DebugLevel.info.build.code_debug=3 +rakwireless_rak3112.menu.DebugLevel.debug=Debug +rakwireless_rak3112.menu.DebugLevel.debug.build.code_debug=4 +rakwireless_rak3112.menu.DebugLevel.verbose=Verbose +rakwireless_rak3112.menu.DebugLevel.verbose.build.code_debug=5 + +rakwireless_rak3112.menu.EraseFlash.none=Disabled +rakwireless_rak3112.menu.EraseFlash.none.upload.erase_cmd= +rakwireless_rak3112.menu.EraseFlash.all=Enabled +rakwireless_rak3112.menu.EraseFlash.all.upload.erase_cmd=-e + +############################################################## kodedot.name=kode dot kodedot.bootloader.tool=esptool_py diff --git a/variants/rakwireless_rak3112/pins_arduino.h b/variants/rakwireless_rak3112/pins_arduino.h new file mode 100644 index 00000000000..5d1e451494a --- /dev/null +++ b/variants/rakwireless_rak3112/pins_arduino.h @@ -0,0 +1,50 @@ +#ifndef Pins_Arduino_h +#define Pins_Arduino_h + +#include + +// Reference: RAK3112 Module Datasheet +// https://docs.rakwireless.com/product-categories/wisduo/rak3112-module/datasheet/ + +// Note:GPIO33,GPIO34,GPIO35.GPIO36,GPIO37 is not available in the 8MB and 16MB Octal PSRAM version + +#define USB_VID 0x303a +#define USB_PID 0x1001 + +#define LED_GREEN 46 +#define LED_BLUE 45 + +static const uint8_t LED_BUILTIN = LED_GREEN; +#define BUILTIN_LED LED_BUILTIN // backward compatibility +#define LED_BUILTIN LED_BUILTIN // allow testing #ifdef LED_BUILTIN + +static const uint8_t BAT_VOLT = 21; + +static const uint8_t TX = 43; +static const uint8_t RX = 44; + +static const uint8_t SDA = 9; +static const uint8_t SCL = 40; + +#define WIRE1_PIN_DEFINED +static const uint8_t SDA1 = 17; +static const uint8_t SCL1 = 18; + +static const uint8_t SS = 12; +static const uint8_t MOSI = 11; +static const uint8_t MISO = 10; +static const uint8_t SCK = 13; + +#define LORA_ANT_SWITCH 4 // Antenna switch power control pin + +#define LORA_SCK 5 // SX1262 SCK +#define LORA_MISO 3 // SX1262 MISO +#define LORA_MOSI 6 // SX1262 MOSI +#define LORA_CS 7 // SX1262 CS +#define LORA_RST 8 // SX1262 RST + +#define LORA_DIO1 47 //SX1262 DIO1 +#define LORA_BUSY 48 +#define LORA_IRQ LORA_DIO1 + +#endif /* Pins_Arduino_h */ From 6474127dd4359bbf3860a6874c9b2b99b871659e Mon Sep 17 00:00:00 2001 From: John Date: Wed, 2 Jul 2025 08:48:03 -0400 Subject: [PATCH 086/173] Update ZigbeeColorDimmableLight to clamp color hue and saturation to 0-254 (Fixes #11527) (#11528) * Clamp Zigbee color saturation to 0-254 * Clamp hue to 0-254 for Zigbee color lights * Use std::min instead of ternary operator * ci(pre-commit): Apply automatic fixes --------- Co-authored-by: pre-commit-ci-lite[bot] <117423508+pre-commit-ci-lite[bot]@users.noreply.github.com> --- libraries/Zigbee/src/ep/ZigbeeColorDimmableLight.cpp | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/libraries/Zigbee/src/ep/ZigbeeColorDimmableLight.cpp b/libraries/Zigbee/src/ep/ZigbeeColorDimmableLight.cpp index caac73b5c68..2fb07fc2187 100644 --- a/libraries/Zigbee/src/ep/ZigbeeColorDimmableLight.cpp +++ b/libraries/Zigbee/src/ep/ZigbeeColorDimmableLight.cpp @@ -1,3 +1,4 @@ +#include #include "ZigbeeColorDimmableLight.h" #if CONFIG_ZB_ENABLED @@ -127,7 +128,8 @@ bool ZigbeeColorDimmableLight::setLight(bool state, uint8_t level, uint8_t red, espXyColor_t xy_color = espRgbColorToXYColor(_current_color); espHsvColor_t hsv_color = espRgbColorToHsvColor(_current_color); - uint8_t hue = (uint8_t)hsv_color.h; + uint8_t hue = std::min((uint8_t)hsv_color.h, (uint8_t)254); // Clamp to 0-254 + uint8_t saturation = std::min((uint8_t)hsv_color.s, (uint8_t)254); // Clamp to 0-254 log_v("Updating light state: %d, level: %d, color: %d, %d, %d", state, level, red, green, blue); /* Update light clusters */ @@ -174,7 +176,7 @@ bool ZigbeeColorDimmableLight::setLight(bool state, uint8_t level, uint8_t red, } //set saturation ret = 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_SATURATION_ID, &saturation, false ); if (ret != ESP_ZB_ZCL_STATUS_SUCCESS) { log_e("Failed to set light saturation: 0x%x: %s", ret, esp_zb_zcl_status_to_name(ret)); From 33c84382635223489d23efd60e39926c3b175f02 Mon Sep 17 00:00:00 2001 From: Paul Price <56952812+strid3r21@users.noreply.github.com> Date: Wed, 2 Jul 2025 08:48:40 -0400 Subject: [PATCH 087/173] Add FED4 board (#11536) * Added Fed 4 board * fixed boards.txt * fixed board.txt again * added usb pid address * fixed typo: updated name to upper case * fix(fed4): update PID and change partition scheme to default_16MB * fix(fed4): remove unused OPI flash mode configurations * ci(pre-commit): Apply automatic fixes --------- Co-authored-by: pre-commit-ci-lite[bot] <117423508+pre-commit-ci-lite[bot]@users.noreply.github.com> --- boards.txt | 185 +++++++++++++++++++++++++++++++++++ variants/fed4/pins_arduino.h | 79 +++++++++++++++ 2 files changed, 264 insertions(+) create mode 100644 variants/fed4/pins_arduino.h diff --git a/boards.txt b/boards.txt index fe67071ea17..4e92d2665e9 100644 --- a/boards.txt +++ b/boards.txt @@ -50762,3 +50762,188 @@ kodedot.recipe.hooks.objcopy.postobjcopy.3.pattern_args= kodedot.recipe.output.save_file={build.project_name}.ino.bin ############################################################## + +# FED4 Board +fed4.name=FED4 +fed4.vid.0=0x303A +fed4.pid.0=0x82E5 +fed4.upload_port.0.vid=0x303A +fed4.upload_port.0.pid=0x82E5 +fed4.bootloader.tool=esptool_py +fed4.bootloader.tool.default=esptool_py + +fed4.upload.tool=esptool_py +fed4.upload.tool.default=esptool_py +fed4.upload.tool.network=esp_ota + +fed4.upload.maximum_size=1310720 +fed4.upload.maximum_data_size=327680 +fed4.upload.flags= +fed4.upload.extra_flags= +fed4.upload.use_1200bps_touch=false +fed4.upload.wait_for_upload_port=false + +fed4.serial.disableDTR=false +fed4.serial.disableRTS=false + +fed4.build.tarch=xtensa +fed4.build.bootloader_addr=0x0 +fed4.build.target=esp32s3 +fed4.build.mcu=esp32s3 +fed4.build.core=esp32 +fed4.build.variant=fed4 +fed4.build.board=FED4 + +fed4.build.usb_mode=1 +fed4.build.cdc_on_boot=0 +fed4.build.msc_on_boot=0 +fed4.build.dfu_on_boot=0 +fed4.build.f_cpu=240000000L +fed4.build.flash_size=16MB +fed4.build.flash_freq=80m +fed4.build.flash_mode=dio +fed4.build.boot=qio +fed4.build.boot_freq=80m +fed4.build.partitions=default_16MB +fed4.build.defines= +fed4.build.loop_core= +fed4.build.event_core= +fed4.build.psram_type=qspi +fed4.build.memory_type={build.boot}_{build.psram_type} + +## IDE 2.0 Seems to not update the value +fed4.menu.JTAGAdapter.default=Disabled +fed4.menu.JTAGAdapter.default.build.copy_jtag_files=0 +fed4.menu.JTAGAdapter.builtin=Integrated USB JTAG +fed4.menu.JTAGAdapter.builtin.build.openocdscript=fed4-builtin.cfg +fed4.menu.JTAGAdapter.builtin.build.copy_jtag_files=1 +fed4.menu.JTAGAdapter.external=FTDI Adapter +fed4.menu.JTAGAdapter.external.build.openocdscript=fed4-ftdi.cfg +fed4.menu.JTAGAdapter.external.build.copy_jtag_files=1 +fed4.menu.JTAGAdapter.bridge=ESP USB Bridge +fed4.menu.JTAGAdapter.bridge.build.openocdscript=fed4-bridge.cfg +fed4.menu.JTAGAdapter.bridge.build.copy_jtag_files=1 + +fed4.menu.FlashMode.qio=QIO 80MHz +fed4.menu.FlashMode.qio.build.flash_mode=dio +fed4.menu.FlashMode.qio.build.boot=qio +fed4.menu.FlashMode.qio.build.boot_freq=80m +fed4.menu.FlashMode.qio.build.flash_freq=80m +fed4.menu.FlashMode.qio120=QIO 120MHz +fed4.menu.FlashMode.qio120.build.flash_mode=dio +fed4.menu.FlashMode.qio120.build.boot=qio +fed4.menu.FlashMode.qio120.build.boot_freq=120m +fed4.menu.FlashMode.qio120.build.flash_freq=80m +fed4.menu.FlashMode.dio=DIO 80MHz +fed4.menu.FlashMode.dio.build.flash_mode=dio +fed4.menu.FlashMode.dio.build.boot=dio +fed4.menu.FlashMode.dio.build.boot_freq=80m +fed4.menu.FlashMode.dio.build.flash_freq=80m + +fed4.menu.FlashSize.16M=16MB (128Mb) +fed4.menu.FlashSize.16M.build.flash_size=16MB + +fed4.menu.LoopCore.1=Core 1 +fed4.menu.LoopCore.1.build.loop_core=-DARDUINO_RUNNING_CORE=1 +fed4.menu.LoopCore.0=Core 0 +fed4.menu.LoopCore.0.build.loop_core=-DARDUINO_RUNNING_CORE=0 + +fed4.menu.EventsCore.1=Core 1 +fed4.menu.EventsCore.1.build.event_core=-DARDUINO_EVENT_RUNNING_CORE=1 +fed4.menu.EventsCore.0=Core 0 +fed4.menu.EventsCore.0.build.event_core=-DARDUINO_EVENT_RUNNING_CORE=0 + +fed4.menu.USBMode.hwcdc=Hardware CDC and JTAG +fed4.menu.USBMode.hwcdc.build.usb_mode=1 +fed4.menu.USBMode.default=USB-OTG (TinyUSB) +fed4.menu.USBMode.default.build.usb_mode=0 + +fed4.menu.CDCOnBoot.default=Disabled +fed4.menu.CDCOnBoot.default.build.cdc_on_boot=0 +fed4.menu.CDCOnBoot.cdc=Enabled +fed4.menu.CDCOnBoot.cdc.build.cdc_on_boot=1 + +fed4.menu.MSCOnBoot.default=Disabled +fed4.menu.MSCOnBoot.default.build.msc_on_boot=0 +fed4.menu.MSCOnBoot.msc=Enabled (Requires USB-OTG Mode) +fed4.menu.MSCOnBoot.msc.build.msc_on_boot=1 + +fed4.menu.DFUOnBoot.default=Disabled +fed4.menu.DFUOnBoot.default.build.dfu_on_boot=0 +fed4.menu.DFUOnBoot.dfu=Enabled (Requires USB-OTG Mode) +fed4.menu.DFUOnBoot.dfu.build.dfu_on_boot=1 + +fed4.menu.UploadMode.default=UART0 / Hardware CDC +fed4.menu.UploadMode.default.upload.use_1200bps_touch=false +fed4.menu.UploadMode.default.upload.wait_for_upload_port=false +fed4.menu.UploadMode.cdc=USB-OTG CDC (TinyUSB) +fed4.menu.UploadMode.cdc.upload.use_1200bps_touch=true +fed4.menu.UploadMode.cdc.upload.wait_for_upload_port=true + +fed4.menu.PartitionScheme.default_16MB=Default (6.25MB APP/3.43MB SPIFFS) +fed4.menu.PartitionScheme.default_16MB.build.partitions=default_16MB +fed4.menu.PartitionScheme.default_16MB.upload.maximum_size=6553600 +fed4.menu.PartitionScheme.large_spiffs=Large SPIFFS (4.5MB APP/6.93MB SPIFFS) +fed4.menu.PartitionScheme.large_spiffs.build.partitions=large_spiffs_16MB +fed4.menu.PartitionScheme.large_spiffs.upload.maximum_size=4718592 +fed4.menu.PartitionScheme.app3M_fat9M_16MB=FFAT (3MB APP/9MB FATFS) +fed4.menu.PartitionScheme.app3M_fat9M_16MB.build.partitions=app3M_fat9M_16MB +fed4.menu.PartitionScheme.app3M_fat9M_16MB.upload.maximum_size=3145728 +fed4.menu.PartitionScheme.fatflash=Large FFAT (2MB APP/12.5MB FATFS) +fed4.menu.PartitionScheme.fatflash.build.partitions=ffat +fed4.menu.PartitionScheme.fatflash.upload.maximum_size=2097152 + +fed4.menu.CPUFreq.240=240MHz (WiFi) +fed4.menu.CPUFreq.240.build.f_cpu=240000000L +fed4.menu.CPUFreq.160=160MHz (WiFi) +fed4.menu.CPUFreq.160.build.f_cpu=160000000L +fed4.menu.CPUFreq.80=80MHz (WiFi) +fed4.menu.CPUFreq.80.build.f_cpu=80000000L +fed4.menu.CPUFreq.40=40MHz +fed4.menu.CPUFreq.40.build.f_cpu=40000000L +fed4.menu.CPUFreq.20=20MHz +fed4.menu.CPUFreq.20.build.f_cpu=20000000L +fed4.menu.CPUFreq.10=10MHz +fed4.menu.CPUFreq.10.build.f_cpu=10000000L + +fed4.menu.UploadSpeed.921600=921600 +fed4.menu.UploadSpeed.921600.upload.speed=921600 +fed4.menu.UploadSpeed.115200=115200 +fed4.menu.UploadSpeed.115200.upload.speed=115200 +fed4.menu.UploadSpeed.256000.windows=256000 +fed4.menu.UploadSpeed.256000.upload.speed=256000 +fed4.menu.UploadSpeed.230400.windows.upload.speed=256000 +fed4.menu.UploadSpeed.230400=230400 +fed4.menu.UploadSpeed.230400.upload.speed=230400 +fed4.menu.UploadSpeed.460800.linux=460800 +fed4.menu.UploadSpeed.460800.macosx=460800 +fed4.menu.UploadSpeed.460800.upload.speed=460800 +fed4.menu.UploadSpeed.512000.windows=512000 +fed4.menu.UploadSpeed.512000.upload.speed=512000 + +fed4.menu.DebugLevel.none=None +fed4.menu.DebugLevel.none.build.code_debug=0 +fed4.menu.DebugLevel.error=Error +fed4.menu.DebugLevel.error.build.code_debug=1 +fed4.menu.DebugLevel.warn=Warn +fed4.menu.DebugLevel.warn.build.code_debug=2 +fed4.menu.DebugLevel.info=Info +fed4.menu.DebugLevel.info.build.code_debug=3 +fed4.menu.DebugLevel.debug=Debug +fed4.menu.DebugLevel.debug.build.code_debug=4 +fed4.menu.DebugLevel.verbose=Verbose +fed4.menu.DebugLevel.verbose.build.code_debug=5 + +fed4.menu.EraseFlash.none=Disabled +fed4.menu.EraseFlash.none.upload.erase_cmd= +fed4.menu.EraseFlash.all=Enabled +fed4.menu.EraseFlash.all.upload.erase_cmd=-e + +fed4.menu.ZigbeeMode.default=Disabled +fed4.menu.ZigbeeMode.default.build.zigbee_mode= +fed4.menu.ZigbeeMode.default.build.zigbee_libs= +fed4.menu.ZigbeeMode.zczr=Zigbee ZCZR (coordinator/router) +fed4.menu.ZigbeeMode.zczr.build.zigbee_mode=-DZIGBEE_MODE_ZCZR +fed4.menu.ZigbeeMode.zczr.build.zigbee_libs=-lesp_zb_api.zczr -lzboss_stack.zczr -lzboss_port.remote + +############################################################## diff --git a/variants/fed4/pins_arduino.h b/variants/fed4/pins_arduino.h new file mode 100644 index 00000000000..f3741700ffa --- /dev/null +++ b/variants/fed4/pins_arduino.h @@ -0,0 +1,79 @@ +#ifndef Pins_Arduino_h +#define Pins_Arduino_h + +#include +#include "soc/soc_caps.h" + +#define USB_VID 0x303A +#define USB_PID 0x82E5 +#define USB_MANUFACTURER "Smart Bee Designs LLC" +#define USB_PRODUCT "FED4" +#define USB_SERIAL "" + +static const uint8_t TX = 43; +static const uint8_t RX = 44; + +static const uint8_t SDA = 8; +static const uint8_t SCL = 9; +static const uint8_t SDA2 = 20; +static const uint8_t SCL2 = 19; + +static const uint8_t SS = 47; +static const uint8_t MOSI = 11; +static const uint8_t MISO = 13; +static const uint8_t SCK = 12; +static const uint8_t SDCS = 10; // sd cs pin +static const uint8_t DSCS = 14; //display cs pin + +static const uint8_t A1 = 1; +static const uint8_t A2 = 2; +static const uint8_t A3 = 3; +static const uint8_t A4 = 4; +static const uint8_t A5 = 5; +static const uint8_t A6 = 6; + +static const uint8_t D1 = 1; +static const uint8_t D2 = 2; +static const uint8_t D3 = 3; +static const uint8_t D4 = 4; +static const uint8_t D5 = 5; +static const uint8_t D6 = 6; +static const uint8_t D8 = 8; +static const uint8_t D13 = 13; +static const uint8_t D9 = 9; + +static const uint8_t T1 = 1; +static const uint8_t T2 = 2; +static const uint8_t T3 = 3; +static const uint8_t T4 = 4; +static const uint8_t T5 = 5; +static const uint8_t T6 = 6; + +static const uint8_t BOOT_BTN = 0; +static const uint8_t VBAT_VOLTAGE = 7; +static const uint8_t LDO2 = 47; +static const uint8_t STATUS_RGB = 35; +static const uint8_t RGB_STRIP = 36; +static const uint8_t INTERRUPT_PIN = 18; +static const uint8_t USER_BTN_1 = 14; +static const uint8_t USER_BTN_2 = 39; +static const uint8_t USER_BTN_3 = 40; +static const uint8_t AMP_DIN = 39; +static const uint8_t AMP_SD = 42; +static const uint8_t AMP_BCLK = 45; +static const uint8_t AMP_LRCLK = 48; +static const uint8_t MSBY = 15; +static const uint8_t TRRS_1 = 4; +static const uint8_t TRRS_2 = 2; +static const uint8_t TRRS_3 = 3; + +#define PIN_RGB_LED STATUS_RGB +// BUILTIN_LED can be used in new Arduino API digitalWrite() like in Blink.ino +static const uint8_t LED_BUILTIN = SOC_GPIO_PIN_COUNT + PIN_RGB_LED; +#define BUILTIN_LED LED_BUILTIN // backward compatibility +#define LED_BUILTIN LED_BUILTIN // allow testing #ifdef LED_BUILTIN +// RGB_BUILTIN and RGB_BRIGHTNESS can be used in new Arduino API rgbLedWrite() +#define RGB_BUILTIN LED_BUILTIN +#define RGB_BRIGHTNESS 64 + +#endif /* Pins_Arduino_h */ From 12e881b6d6cd91bfbe0f58feac10bb58ceeb34c2 Mon Sep 17 00:00:00 2001 From: Paula Scharf <48286621+PaulaScharf@users.noreply.github.com> Date: Wed, 2 Jul 2025 21:22:20 +0200 Subject: [PATCH 088/173] fix(board): Update variant.cpp for senseBox MCU-S2 ESP32-S2 (#11532) * fix(board): Update variant.cpp for senseBox MCU-S2 ESP32-S2 * fix(board): translate comments * ci(pre-commit): Apply automatic fixes * fix(board): translate comments * ci(pre-commit): Apply automatic fixes --------- Co-authored-by: pre-commit-ci-lite[bot] <117423508+pre-commit-ci-lite[bot]@users.noreply.github.com> --- boards.txt | 2 +- variants/sensebox_mcu_esp32s2/APOTA.bin | Bin 0 -> 783824 bytes variants/sensebox_mcu_esp32s2/APOTA.ino | 283 ++++++++++++++++++++++ variants/sensebox_mcu_esp32s2/variant.cpp | 74 +++--- 4 files changed, 331 insertions(+), 28 deletions(-) create mode 100644 variants/sensebox_mcu_esp32s2/APOTA.bin create mode 100644 variants/sensebox_mcu_esp32s2/APOTA.ino diff --git a/boards.txt b/boards.txt index 4e92d2665e9..4198f3856b0 100644 --- a/boards.txt +++ b/boards.txt @@ -40993,7 +40993,7 @@ sensebox_mcu_esp32s2.menu.PartitionScheme.tinyuf2=TinyUF2 4MB (1.3MB APP/960KB F sensebox_mcu_esp32s2.menu.PartitionScheme.tinyuf2.build.custom_bootloader=bootloader-tinyuf2 sensebox_mcu_esp32s2.menu.PartitionScheme.tinyuf2.build.custom_partitions=partitions-4MB-tinyuf2 sensebox_mcu_esp32s2.menu.PartitionScheme.tinyuf2.upload.maximum_size=1441792 -sensebox_mcu_esp32s2.menu.PartitionScheme.tinyuf2.upload.extra_flags=0x2d0000 "{runtime.platform.path}/variants/{build.variant}/tinyuf2.bin" +sensebox_mcu_esp32s2.menu.PartitionScheme.tinyuf2.upload.extra_flags=0x2d0000 "{runtime.platform.path}/variants/{build.variant}/tinyuf2.bin" 0x170000 "{runtime.platform.path}/variants/{build.variant}/APOTA.bin" sensebox_mcu_esp32s2.menu.PartitionScheme.default=Default 4MB with spiffs (1.2MB APP/1.5MB SPIFFS) sensebox_mcu_esp32s2.menu.PartitionScheme.default.build.partitions=default sensebox_mcu_esp32s2.menu.PartitionScheme.defaultffat=Default 4MB with ffat (1.2MB APP/1.5MB FATFS) diff --git a/variants/sensebox_mcu_esp32s2/APOTA.bin b/variants/sensebox_mcu_esp32s2/APOTA.bin new file mode 100644 index 0000000000000000000000000000000000000000..0ea39335dce0e13a3c900367b76eacc9b413922a GIT binary patch literal 783824 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’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 KyFd;c3iu+q7se)RIqiEi%qr+Nis-^Sv9m4z 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

mA-Yt~G^OxjK{C27_fn`#R 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 Date: Wed, 9 Jul 2025 16:52:51 +0200 Subject: [PATCH 123/173] fix(board): add APOTA description comments --- variants/sensebox_mcu_esp32s2/APOTA.ino | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/variants/sensebox_mcu_esp32s2/APOTA.ino b/variants/sensebox_mcu_esp32s2/APOTA.ino index 5f683752abc..afb741499b2 100644 --- a/variants/sensebox_mcu_esp32s2/APOTA.ino +++ b/variants/sensebox_mcu_esp32s2/APOTA.ino @@ -1,3 +1,7 @@ +// APOTA is a arduino fallback sketch that is written to OTA1_Partition. +// APOTA opens an access point which waits to receive a .binfile on /sketch. +// After succesful upload, the file is written to OTA0_Partition and the microcontroller reboots to the newly uploaded sketch. + #define DISPLAY_ENABLED #include From 76210797ed11288bfd3c6c27122f0103562196d3 Mon Sep 17 00:00:00 2001 From: Paula Scharf Date: Wed, 9 Jul 2025 16:53:33 +0200 Subject: [PATCH 124/173] fix(board): add APOTA to senseBox Eye --- boards.txt | 2 +- variants/sensebox_eye/APOTA.bin | Bin 0 -> 976464 bytes variants/sensebox_eye/APOTA.ino | 287 ++++++++++++++++++++++++++++++ variants/sensebox_eye/variant.cpp | 65 +++++-- 4 files changed, 343 insertions(+), 11 deletions(-) create mode 100644 variants/sensebox_eye/APOTA.bin create mode 100644 variants/sensebox_eye/APOTA.ino diff --git a/boards.txt b/boards.txt index 2363fe2a63b..faa7f94a994 100644 --- a/boards.txt +++ b/boards.txt @@ -41448,7 +41448,7 @@ sensebox_eye.menu.PartitionScheme.tinyuf2=TinyUF2 Compatibility (2MB APP/12MB FF sensebox_eye.menu.PartitionScheme.tinyuf2.build.custom_bootloader=bootloader_tinyuf2 sensebox_eye.menu.PartitionScheme.tinyuf2.build.custom_partitions=partitions_tinyuf2 sensebox_eye.menu.PartitionScheme.tinyuf2.upload.maximum_size=2097152 -sensebox_eye.menu.PartitionScheme.tinyuf2.upload.extra_flags=0x410000 "{runtime.platform.path}/variants/{build.variant}/tinyuf2.bin" +sensebox_eye.menu.PartitionScheme.tinyuf2.upload.extra_flags=0x410000 "{runtime.platform.path}/variants/{build.variant}/tinyuf2.bin" 0x210000 "{runtime.platform.path}/variants/{build.variant}/APOTA.bin" sensebox_eye.menu.PartitionScheme.gen4esp32scheme4=Huge App (16MB APP) sensebox_eye.menu.PartitionScheme.gen4esp32scheme4.build.custom_partitions=gen4esp32_16MBapp sensebox_eye.menu.PartitionScheme.gen4esp32scheme4.upload.maximum_size=16646144 diff --git a/variants/sensebox_eye/APOTA.bin b/variants/sensebox_eye/APOTA.bin new file mode 100644 index 0000000000000000000000000000000000000000..571336d378a9aaee97e074142d0e73cad85eacc9 GIT binary patch literal 976464 zcmd44O^js8mgjZrz3S=ieEpgkji6yPPu!{@t0Sv2KJ(*M1y$AK;o*_#o$=w%-J`Oq zA5XVE+&v<^)5G1%KQbbpT7v+o#RxVqAU3dKF$=^Bu|^2Nte6c8#EJ#5m<=P?fP_Tz z`yVsAAO4k5-7TO>8Gf(bV_(OP9XodHoMZpwul?xk>GjY5?*AG^(QorF`tv{6-w)#G zCtv=X|IUx1z2<-WpC}>x*J?j~*4^ub*M;BKxHIbwhMWD~#pcDV*Y9-41(=|- zPk$Z~|8O>l_r8etcJ`iG(%WZSPq+3q_nvKDeg5q8))$|D-g({)!2JH+&foeU{=@ih z|J(n~fB!rG)j$1v|Lm8a{=@Hn_iz7$r(b{fljApkF!=i4i=uxN{q-FDU-t1}IGo0# z)_B^R_J)JFecc*dbvyBN7@svuJ6mzRJDKY1;A(4YDjbO#?rlkQ;BJs95YN86J(-6@|N_7mUs z#+@jB7}dM&ZtpF1#^<9>Yueq9A5I>}joz>I)OiqPsSl#^a4_h$)%d90A5OZ_w0k$* z9`#$jL3B7Ajk^8Lc-DRscgJs9gTeHp=yCbwe)RaPRymCxzpm8yQ!6F4#w&Z+OP>;w z!&{}~z@DUE_w+fumTFb^;p}zAN_mo0PMb;NyxOcJ2g&Xid&xny89i<^OVnI%G^+d2 zaXoshVm&=2-M8JrbP~O#E5q^resYqGyRCkI*iL#NYdnn}?(BWLzv@pEe|cWx&*9nY z)9CBy@MXDHj*_#ZBYSK_k4xvxGaw$nDAoB}ttBUChtcC&jhBOZ_3#Dl{^og72OZt% zb$1*`y+L;xM=^hIT9Y?CU5tmVPP;WB^@<_6ZM`G0GnhEg=e$hcjkKbQ~R5 zPhTa+XJsB6LW6reKdt`aJiJtE-#krDO0`<$FyZ-m)SZl;>`k6*Mh_?IX!sZPsWrM7 z4*T)z?nR?Je%lp>+QWVyY`yG4RO5~Ia6r=_-|?r9i60$k?Ho`=I!n6=H+mFvpF00yMy+y z)9pkzv;MTFiMX8>cHA9YO|PSuL%M%hIj%G-(ThqmdRZwQMz!;1bXIFt&rTarvtBA! zqVieoyXfRq`Rq74Is2v()#_)pqv|Q3Jk(0f^2_M_lwo)k)hh=|KR>mfQu&ya+F89x z?ozo~{ifRdE-JsQlwX}u$w{RSb(}ORr8*^?o;9mS-;uR(e$Xh_s|VC`ZUCdNV1 zsjgZW{OI8w{T5z&gIPCf-1eC0n$orLa5`l0_y~zl>3XT%?vC_uIvi}4X4C88xHkn? zZ@Y1IAUqk~_!@gO!yqP*XLw^YroC1_eLpc&_PcTUdf01sC(+69Ej+E(rKAC))PEN> zy50CpluoJB_+;4WUG`|e`J}7%-Mx#N-J8*nv3~~#?DWPET6ECr@bdi(ju@Q}TCSbH zCReTX?uKIRBcPq=Xb3ax!0W8^BXP*+go!om#Hy~uyuJm@o~`e#i+(qPM|!pZY=(6^fs}i<`q?eB2se zb;F0v^LjOxdOn!UMj(NaiBCX6D{eBw12BnWtrf8rfe9ltx$IdtDt9BWWkVMGimObH z@qiI>lQ*u6$3utt>dmO%74m4F@cY8C7RG1%mZ6Km^IIBk1^aIq23|IswU~uxqLs^< z5!xH+)AsI`kYw}HTJGp4)m zjoIeexOc_6vhrR#Yzi!B<-D?xIrt@Vt6@mb`qOln={E z+G_mTxN!qRc=GT*UKe=u7rpa{=BzzfXA3CjM(h{Y`pM0`3 zIS))JDj>SwvN2^@@=M5OPBxru_d1uG<8Hsppd}LeHiNjg`F7W>F_S0Ty+OM_>vUOs z`b*Q))qa2LI(kv7o<--SdNZki>wcU5w+u$y2Ob*L7bhi_ie@P)fv~gFO5^2O)1J#` zr_JcNa#)V)C(VT1`k}DuUL$EV50jJf@vHRZApi2Zk)|JNMF=?#tEcwdJUq#A>m|*t z7oS^?X>NP*xodoAew%~}{i{HCtn^&MZkkXpX5f{^Wbp320M7Tle0JWS@Lom$X*4VK zq;Xg(ys{|jYvZ`|O(p&4-}u%*Vzc(1A!;_P;_1Hh19Qw4w zo$#<59`?e+lL!$`i+JN8IXrPcwW!j3S*f2^kTz@82zgF|T}2qpy)eqzmpWvUy@`4* zW|Pr$tf}#FGQOD}cBifO^}%cs^C3PxKR%8}!=5ujf4;-sjU_i%8r+e3qc|2{ z)9{5NRcQ1JK0kv*U+U!pqG7U69Q2wHN_8=3fU-2M54zO*`>@qWB>_i*R)Zy(1K zr(F5Do5ay$+s*ICG23%qCbJ9e?)T#(Hi&2S@A66Lgc2&vCp9Y#)E!waRkHUjDXx&i z67oszv(!pa?IoMevwG$5{G=A0e$z-^RlaLPm80`UC8@C7ogP-$%~lRI!6e-uMn|Ql z7S^Mq2HJ*Z^SHu8V|Y20U>bV6`)vQo?*5aV=nrR1Sg(iE?6?`RC5jK>J=>Cojp&S4 z`)pV5Ad~>2-FJM}0Kwqa(Q>1TcOK!3sSLA3mWCdp#U?X+u@!XR{mmD)e?ID?sWGzn>WjVI=Q}%(5gA1DUtQm}JMG7a z(q{wd{l4mSFVTujAD=YlQx1>L=U`m1R!qnSClMq0YVYaJW1Bqo_50q?y=m@zwfp?! zak?*Pw6EFheT7Eg-RyGj@nQF3c6E&AtN)cYV~>qDeAVeqwB74GK4#Gp8~bYaamBP_ z;YAxEtIuWo!)ED4snbG(Fb+M+Vf5%12dBbkz*~;3e^q4rl|d)Z^x^tB*tm2_`KuNP zU?ri4!Hr{k+m8><8%guKTIH7r3EQ(lZ~CiWO2hgqvBdi=ym2hphfik^MD)vtvtQ*K z^;Mx&^X;cmSEJrAs=%-Mvqw;`D0t&x=aHjiYa?RB1^secvW1j*!$gVA;h_Hxy>BPJ zZcW%}$0iPHLti_8~tDeTD zptP*u%DT{-jAN9hra|bb!Bbj{9vg*W2kjugu3(>-0H1jHjYsiiZ_Kvb3aU2h@nF{P z7jVd&OM0WWHl6(hC13lE)LNIOV~IE5b$7@>`IcG+32w&x)iTHNZhQ}4XXnp`N% zz;9b`x(F&ZEM7@7iTj#K@eLFGF>>cL?siy4^a?JfTPZ5{AI=^`^=6q3WxkPtGVBYT z&dB0{6tfAEUosp`3VDQB(`LkcMN!`>_WS0hMH}50unO|Gq~!Pt6n&1oFuHy>Nrsn~ zllY^rVii0^DU>!F z42+PxZbhYDwsaiu4qPFTI<0a(3rZuYaNH13Y6=xwZN>6aF6=&9&YRu@%SVAAhRrqZ|1jh6-~Ee*B1?}-fnl>3&3E(lGgZQ%NE;;+AocQoJ^JbxKGA>!cp6k(t0z%vJp5SMi1^JasVBNhCeF$v^oewft(U%j&V|RH7>&jG7+B+a_xbFE=kudyI&F3{MAJWxtc89aUxz< zXw{TsBRfZ(1B$1&D7K5$Pe#4uQd=e~#xD>in={>RI8RDACepAz*;^#vT={+}*v=(c zu}0kUiO@Fa9_EU+*9$h{w&GIVQ{((S5lf+OrZb+fD!WB*D=xSCZI0Kpkr$7EvXB)V zx5$r_l@zz8`qaNOVWKy9+v@i^P_}f=syyC+((RE*j@j2KJPPl z`D#N~LtA~}hKnL(!}o@Mh`em8#$klered|exCT3ZU_8QqD;AV%rwCNSiL%=df{uwS z#T*W=T9r9qleBj`X3gp{Uvgti`gNge4x>sANZ5>BNw5v{eiHGhC(LEbj~v`#MGPzU z0;A8hJ3eeBf{?#g2}?sd6E#m3=S@24wzH{6 zQMPV;9LCBu@^Q=><=Y#iv)KTIf|!pBi11P%@&k*JV>x2O4hlVx1`G3q;e3UV>SscCtBm^C(6p zKvEvF&a?)$*yxichGZtJJaj7RM)#U|&f&XqZRy~Y(;m+fic}SZ%O`s*R4F-MI%N`Q z(h*dkM)B!{wQ`Db!^U)hGP03NpOWDuxttB!68gXxFUC%XqvU3WLC9((~bf?-HcV=L{LC?GFLdd838wkUP=f|8-aP-bW4M#wgLp#Q@(wRLwkD}gCivx<0 zNM-; zm54MB8(IT+xtlL+Ep>DNeK`XMX8aU=%VaCI^FKt^6c*uE_C}ibl7q1~`YHRl*%;L% zr_cUGnf(LyF{a)`1!}o}pTajH2rxo@YgsQ?f8N-kg3miC@KXMIy-HbkpE375%^YOh z=Iqs4l$v-rW9CD~8W^L#+s?EwTf;=s$M{yFduMpX1L7gOzcG`PT>=X#W(mt@pk;(9 z2R4RgZ_*z3(AzGDM;NBm`@>uL>ArAHnJZw7Dl|q90#o;twT>DCs z2(|;vGqxP;rr;u%9Hoq>us4Id9uD6m994+9xSW>KzL3$=*aWvng|3G-R=u3XF3Aq- z=Qf`K%(@b2n4HK241Q)z;H_8BvSn-8y6-j}@C(fzS&B|E zZ$#cSHn?edyjTON=PM}NYRK9P#h#rWRbM1W^-87QJZmJF&q+nD61xLal4?~(@SVA% z!D2z_kgE;0;d9G%db$)XH%9dbgyMakeO7JRizn{}ZRr?VH^>ZQ`mN*089R8{-FbrnImi@B;x*Mi#_}+Y@#pguc~Rzmd6xD4l~2H zv7kui(V;7SX{*K;j<{ zkWjnD4C=5!Qx!p~ZJBKog`S+8YkfwZvx5LXa&zr-&^9G0BMd9UC_S&=z1_@QIgcV! z>f83qEtxFEli`51wPCZRG3}1x=;iG}wnSr^q`tOWCRp0Uc58Ra8nnE}(Q7St?#oWb zHsa`@UManzZxf6d+{sJ5u@N5-(neljzQr0RvDWj(wPW-7Jvt~)eP zM`K?1imy+KuXfm|Eqhce>q{I*H60r1p|leotVpF;VS2D)^x#e_d3Y9ppNEH^hlkI@ z!x!OU*A*R=c3sgtxT1M*Mf2c_=D`&mHA;2MA633BmrES5>EWBw;bA>GKB&CVH!Out z58Xrg?EJJzN+tzK+?WmAde+%W&#K)>^Xk9E9i8uFq<}g0Ogd0)&5bs4pl`4bM0uTk zDi25Fp)By&d1a{#B~wV}1+1RN_cwTO zF4Y=H%5u0YRdCJYgOlFG6|HAk&%W$Utt#tX+A3GOPa9tM*v#cy8VX9A9!k5Wv@|g* zO-XDTLpg@J+}$q4b;!3oOJ(vccGy3C za1ca2X8)glxye#mKd-T!@^7cj>gjo*kv!cwS-W|V zneO&v*w)5OHuT$+s?D8bLeqrIixQ{jCkJdQ&yEsF6pgiox5pf8xS|Hquy@Df0$wC1 z#@e=+AnXMbIA0jZs9{u}6&Tu6u9p{p zzp2z4nBXs@pOi9KYf$yH+F&D}0xGd|e2l|LE)AoHgHo<$SwRSp%*`q|`J*}DC0!+l z&YKgg5KqkHKXgxJu}&Uh6MFnz)(l@m<@D_Q#Y=1m@mzy3=}4MqXE=kLW@D?C)k-hOpo6<*l@~`{@S$0(QZJe>3z?uM$;(zG_9KDD z$OKv8d$OW_ENhpe^R)6h@xwe%SCsi}vw^QqxxTEo`gT}Z{jGe2XwM1f9FmkxpO!br zD$}P$iWXvIG_#gwJkt{(D5_`luiUZ`mgmwTMm11&0ei3O44U=zJ99fpKb{}gie#$< zO-V!bK{Xp`eMm>{3|cgiBMgzg9-h~Zt7YbRa$Gv7WYnN5T*R|Wa)snK|SR-a3qUw{nsb?KEOs$$b#pPe5cCN>ns55j{Q3AGkK)Xp%$3*=!% z3yD%7=N3@c-sTzebunE<)9|xwB{Ohr^$)8JTf2(oH(#^x=1I?iliH{>&uhuj{wkCI z{B40o<}%kSf4I;^no-6>q}Yl&VpH|h=$w^&ZmjB+7nm;8@lh*a=5@`jnv1E8Oe`VC z>DlWdf%>!s67}hM2_^+C$T~i?fVLEJUXJk}s_^LgDSnl_nYOn#~M_g2%n|<-!2#L&zbMk+jwmyi$sYv=kfibYd5w z7f@_Dlq&w6TVv96C_1;gWywwuTr7QF8uK|cTrzkK$hKq#_WXo}vY5&yx&miBX`UC; z=Lz465b?Tsn`2b=1qWHgl)_}lr{`ywr6cgLnrYcCioM}uomG+Y3L>$6bMB*w*-I6= z=RcgEzJlU2VzEr$vSd=@YaB?=UKjgmgo%N0MoN~$Pt$B|FXoz`#g^wA+)Q&TldJh8 zw+L8@!<3}jVv)6M+8bQGQvQYo>$s8^uGM4wW~xU;O3ac2>4MzSNmgPG;K(E`)t}Kv z_Su}U-HF}&f?ot;BVc8O)4jAaE5VA(TLw0_lgd7*E!Nlu%s#&~=0NII?k4u8>nFeY!~rSII^u13EcAe7>fF+H;c&S3{^X@>yekuLklQEY|=hh!iA2 zm&Ec{m~Cn5{KR!10xQyv3?V5UE%#NP(SEC$yv(n`i=WFctN3quChk5Xo;6wLa;3`D zy-1;!f*~6eeOQIGGcIIb=A{8kd|j6hzS-S}3N8Jpf?IeZhw_}LYF`5Nvs#|AA3-= zg4Pyze4{4uJ+#Fij917V+=J-3EEi0GQDe32CJvpo&yr6}wN^?^-05uxg zsx(vVu7h)vnl0^7>HL^Y>Y5@23R|n0AZEy33I6S~oiCEI>j|zN?<0TXtvqUtUv!^Ir=DL?AA2yPg zY=Uj5mgM4d=nF)&9^kP0LY`xbRp$z+p)|@bJ2YPmz1X^t=!AEt67{V}OAjy&R8H)4 zIsFO^T?evM#k8cTb9QLe7v5jZ@nzpo4)IT3;CpYo`IS|oFJQ;Ns?EszrIjB!3qyS# zvN|&)Jgo~V`=(jT$2xLwexMIK_AUu-*R)#O_MO>>Z~H!0NS$DI8LWjd7B*v(1Yi6`ELjiCbR^&Zv@V_T zN59R<%fpG;T42v|hdIm$qv?;F4J-zXSVT@Y>}`o{pH(pJPcFCh$9XoGDVol7CPbjH zWDft65fLx;G08J8a&xG`>JyW*E7`?eyo(>s%#3>i*(}dv#vmc%MR$7J#kieZG8D;W z4rW@4OCTetSnegV5|HJX^Q>WjI8`$PhL2a^EbEdE_le|E=#<0FbSR{I)QHlpH(6Rz z0C*hR%r+JGBl9YbBUtI`Qs7o#rCx{BLBD-0>zqb=mZ8n}+ ziOB$~1z)&pk`kQtbL^`S*3xQRtIXopJr)YExvnTq4=U6UeT-c;?FmJ$g*2LJV@sg0Z!4$_vzQ#AWW%Ac^TSLy7+EoULN=1y z&VB2{1UlV2=Q4xMHyJl<@ZwLxgIJqGrqQbVO3f|8VcfzbvGkhWf`mywk)NA7*OLK) z+#)$4#cp8}$2*#X5x;rnaat}wjJF(4<_F=CJa0Z#QzL7yO0bPi(8!gvLtXF!X@o__P>CAwNg_G*2gc`)pgouhUgK-!x%!62djhJHfJ%X4ExR#_dp*VEP>1wbtTvtRP zU{z(a6kABAtj5W#eVuQqarI&`y&L}#6NGv5@EIz&IX7z@*NgM=Kd!e?5x!Ivv@a@{56!E73@qz3;7G87J33HF)DSG z7+xOLImFp1rI^8y3!~=oxaqMNYbx{+NSxK^+}8)up~6|# z4AFbUc<9>%t0Rphww*HATD9Z|0x=J1=>l(AP$8paR!R6K?hlxNCLR84Y`h&}Uh?Uq zj$z@EB*Yw`%v%J8Q7xj!9PIl_WWerOo*<(Q*97?Y6Ju>Vq{t^TdtX2k{g+7=_BLi! z2v1*uqG7+|xYpM)3-&fO!N)Ybe;I3}5x!CSFy83ZSD5Tv-zLGK2@_*bdojBV!?#GH zLT4@qkI+kJctgIAKxt?5Yb5-u3H|)0Qa-FS*odLhdC3mp&=|`qkW-9Y9MlqNaIR9; zHSfGmt&_K%w6)oX0?A`MWz5UgLo?U2FaoekRz*bAEPcZ@9`gPcS=EX|=uZ82(}WDU zoHAfCbPwEy*lOyF@O-V7N6vo*H=$mk#Yp5gt+tCo01=QEr#1Yd%o)-7ozlJyEg0-W zrjz(#{JAHyrH%=?5u;r576x^BGw<#ZLVK)@bNL*t{D~QL-;Aa>d*{IbNKI}x6g~wP zhDRntz;I;@F9UG3T;|}#tV%{;hgVIUd9||Xw5JGW76NKWUTRaxGugMZ0FjFbx5Z2Z zEwsk7uR$nhacf}_%A&^$B^6=KA`xcSQhdc&N)oHhH?0Ue%f*}w15$*@BP6%yG>Hp8 z`R=$kQt-bTO0C*UMMJ}RYK7l-ab~%!Nqai(i^}`mZqcE)?maSC_&aE!-kE+?q(HMIB($ju7l<#o zOk3eI6a1ArNgMx+z^}A&aC%bT_}E~ymf;aH=Tho3#zcJh`2GCBaFHm~gci*_`5uBP zf*t6usT!aU$Z6dD6Z1J_%S$9jKGM(&%a7m7rCR$-74|D@bD=z~oC?dwncQJnMFo+p zlMZMzcF3hvBWolYH0iV}<*J3enKW zMMcH5#z~IO3SEOQ%1BfZO60CCheLq1)kuZ|8$s>M9{iB?N5X)yPzI(AYP0wlKd1wq zgR=Bb$8(HCPL@Z`7JVG>6#wAplx)+P$^cXf(QO$aAsn zUhf*X0FQ`o+>P#DvW)k;EaKEKdRxV-vN<#;+l&TbX@|NrjKDZdsb2!N>aBt2#nzo$ z^Q_oAtmxm>d^~C^mU0Tvdjei4-=Ra@-^~>!->ns%s_H|bT*Hwo_paH4KQ@ON1QqGf zH@Z8KV{RC^La#*U9xtKKqxMT}KKPP~k;^c%l_7Y`Rqzto4Z%)`DxegBF<>-){B zn&kuf(RL&X^LbY?9|gICtEv32ASIt0*+VMP>~68%(H*5k^WMYCj|gsM2y2G~|9{u* zh=z`($9p9Foz4_x=oeLT7pA$+LQSbAiavh zim5p~pAT%uOcUwN>@H*Mt5cB(#!kgm+=BA z{>p)h;*9BZCofS_`ut4Y_veS@N?qug7~!NC5ub93xB8`~;jW{*=Fp|^#sCrs8_~-U zsMl-_G30%snL(pBzs^Web3sgZ^Xt*ubV*!bND-`EZu-Vh*Gzq_)d<=aI%={VID#$T z$cWh|c1K~Z7KS-gW4bpBjj=#e8OWEhmXp5<>e2*9E)GR6CBVC%AG!Y^1sie%-u4^( zsCqbIA(Epz-WV~F_3YTXhbyOWpx&Tt4y7>j3p-G@tXJjeP!J6^MTY-B!yE0nt0Gv4 z2Radh8R3b=ajpD1$Rsc(_+mVqjT{EnP}98lg<5+sO|Fzzg8za@lof3V%vD7zXLko| zq}`O_!UPl;J9n3tJNDoCv7`7M=2;O+0z@uznE>mnvSw8vU9KqJcVBggAAqwUEYc*U zGBw4uD}&CK#5ztfY^_JuWf9XC%~yK&DGhehd0iRtqbT68Ap3{JSS^S@emMC!oV7?1 z>;hMyiuXQLXoqhvc(4lbB8>>lU0kAVYhhZ=(zyI}^wUqVDLSk38<*mVIqFI?d5M$0R1Z_NH!_E>3mxNyW6iB%_wY;fha&qi zrb{*8`d#xD)0W3vozV{4k&Hq#s0MjK(-f!v)X4`M?3oLq7qoQ`M}#H}{`d@w(Yz#7 z3!&mx#^w;ogR@pJzZw}uE=h!;9Akv6>+Fzm-(SXsY#k~vnK@C@Q^;lKpA9Cs^}pdQ)I5A>qptYEIy_^u%nw0eT!B-hC7U?G4e zt#NiV{xA?5n^(A_8Voh;B5T*&pX;gnbKk{zx0zI10 z;Wi>#EUsY{^BauF`Ai#)rF%q&Wda0(WVpZv?6UoBxTnuA81X>3V*OM2WyVsVs2t5li7rHKVB zSSAZkkW+VMqcRt_r|eJOU3?JBCKuct#iZRB-PPHJT-Y9$Nn+cspc`S@IYq!6)M(bO zYwOJ%DQ4$`MYb?WZ7$O+SJ9C_o)GC5_e)P%r&i!i(UoZDw*Li73zb`|P--rGFiio3 zx$wbqNC%Y10^Mjr>?_KL=P!VZ0WE@;+7?VxD$9i&D_ayYt!k8$EsI!#&CS|=m%x(G ziqC{n{T@$_-J%joixlSU~bi2fz)}bFlu2+ETEWP{V@E684_~c z2Z1y{fZI5{e~?DZQ!Q!hDKG#n3^>Si^bsJ|5^@0gfrLAc<3Rt*(2Uq+q)9tGCl;|f zq$p_a40cqJ$a?`73SOr;oxs zU4`Xku&a=L=R`=`x9Wu2zf0Lb^Xgd39-JTfxBvt|&UXg|wW*%xWfzc}XOc_#x#m%k zHEH|LY1TrzW_Fm`%c`@zzDC!&U^|z5{gN`6&HnrRGPbmBOPk>KMER1X63Y6ymZcTY zW-Vd}v%CskqL|8RYS8&-UkeL*kg@E-9-Vgkv&*9B?#xz#r86-TRS}6_{45D(s7f5HT^Q6@mE^_ zLuQ}Eb{(2+V&tsMC1@*z4A=Zb1(=AYhXoPTjf0b!%y4fx?v&TrldZvq0ojns0&saj z7h_^qW>+382g0x?Z?m#^s6W0-T$8a;?z6u8Ms4p#+k4SvCp`@2@yqzhqL?h1bQyY* zx0l-MupOD+{qnN&E3}l|F8A+od#Yv^MkSS~ZPK@%Y`dXOn~Y1=Tv)@&#>eUBkF}9h z(KZ%t6Z*{&P7~UL$~C=O4tq;>kKC0zjfkpWm{hjUHTx2_WwC_|#LmeDp(*0fL1M3! zLSM+3VZed&Qh2B2(hXV|cw=az=vRvf&!AQSWDPV>c9zWAW$kmeuW=6Qd!2IL6V^6C z>tIb|qrF+70a^2x(}I1ssIT?}32+YH8q7u!mmW;FMAKWkL@sDBTzPI&J1z`VFrL?W ztgLCd9l?MJc*iJMh(v4_+T6$5)-dJ_LY57qI-&u$&J>2w*DXRV)IS^iWm+(JJBhfl zu25oNiKYkTdP%7aU`}>9D}j%~Z64SRVd%}teha-8Y*D>efaYavAC~6}nha-S?mlr~;IAW$0%H%0xUf^TOrT=q?(CU0aha+i&MaBIfSMLi^a-&*^;N#c3?6 zV2PzPkmpQPmUdg3WErFk`&__4EVkunk5mMx?*34WI^6c|x&z{)5Cn#&@0JD@l z+{d15oJFb%L%VfLoIUC&#yJnjGCgir3p^`~;_}HZRu}iK!?4dZx{>xJ3;eyT^B%Nw zqoAseSl5*tV5BudJ1s4Vl-AQZ%ebudgz+1`i^q2hrH7JO)Sd{lAKP6!+|U?4JTayB z;Yo?mW_DM|as&l%-y*M;folJ7x#AS zX>Ltemmy4nfJb+8J1YskXu?$@L%7Ia3iXwiv4mJ%GTA&nBMgLvI!!Pg#haAC2sZCc zvT!-_+;RDAr+8y1)09e57I#+XS{CZg!R`^$McML;kYR$ z{j=WL-bm}-RF;s-xO8qFWj$X`14YpHOyeA|$-pb8oUTjrQu4};6@}Rh&RV#lxIm@= z-j7VvSu3prjg z^U@LY1J{jlr$Y6!ICjfwhC1>U%e+AQ7nHDmb{7#HI>#ZWtg;Jv?asaq7X}8lu!bnL z*|K`!58U*J@9q`L&Ey_A!*V0q(%ClpAxT}!n;C!`ho$XCnM>8_-#U1W1C&ZUYIm@- zZX9yNtu<3czYlzZSp@zqZh3tlu#-8s_&FD+e_fta+p~nJ=dTQ!1l#B3_;DOD0~tO>BOhBijjnh08S~UQ0%MxB2JuzJHyRf4E(o zFvp4lQ3o!Z8`=9UM&_@>F~2rp%u2WKNDUg6qww{3h&IN!Ae*JxM0cc`n|;`RS#itr zT79C^#mZ7{AS0jnkA)s|H6BIiCntZryvQ(9CT&7yhX@r#(5AeSe2p1bU%P|OQVThQTa+_fNVtGTE z=3!QlmDfS^2A8PbbWfU)XiEC!SWvAt4Q|*;2L%R~>&o}R57X+5Fy<}?1oz4Z*Tsv> zqu4KnTq)+|0lN~VcNyPdoNqQ2EYJxi<3h%Wb!QU;MESGQdvn|4wqo(Y8QV{KuqB)f z2L&M^aMxZ(dp_IC2Th$BvU#pMC&vU{oDJ?{M$gHWq!m%I2weOgDD41B(KMBG%Hhah z%AK73IXA!nt7f}16^%RLEObwsBctVoYXCECVmh0`fxxn5SXm|=!W@XIz@m#@@IG4B zWW9e5x1g9%MyB;M#cyyHf$jXjg;?N9I(I0swRj$}ZhJGETCGILc)? z$r%ahN+ROReb6M1 zj7+pRXq>QJuxC6&4I%EXUf%4C&|gcxK>@5W?e`;BwJxmG`!H6_`g|xOJFGL`SYttM z5}@%7viHNCy}SKYf6@zU*8#cSsh$NCy4s{(vl$E$1mrIH|iwQ31-!hBrDvuPDHnNA_UYy`)0JEGx<{uPowdjaB;!2P=-KnJ4Ehq3Nx84`htSshF zZTh#p(YF23RjWA3>rN3BK%qW=`+_Da?vIXp6^wX3Jsbg=V3@QxEnd!DHZ7Zjfx9T1KmK4kpEq6(B8(*tP8f@(y=FR& zY^_JAce?Lst@3Smh&R)p@ z9c|eoe#t)Chs7*#P8DhBBF9qh(=?!0d%FB)khzPYL_Cp%gr!5!G0VR4ozU zc#(2}jUVDE#G`;|_pDH3W`#*cw4xSE3Ztm_YzWfmxs5h(A@38fEqt}K`_yWHRC_VJ zLOvZK`|sQQWU@@TQeKlyhp-md%abp2LDUFc%UA+0{v_CcXVcj;1Sb`kJq??Qfes9N zV>UIEu1Vob;(Ty8>RpsTbvOHjvL9Aw{jgE`4y5w0+#d0386hg<1UFb;-h_p zB&&TzYS>OuD0D7mwe3$W3mOJ4X5Y#z`@JSiyV1R8&s?uktb}oNT+a14?JYZ$<$YZa zp)Nl6SaS@`5ZCVw-WZoK6v7AR%Ur)w8n@DDGhsq&^_qx%NA~1q?r8ICR(AQgM5FfR z*KP1QBs>zcI*m)X&~0V6bs(zAjFUOxU_an=arhlI{2oA*NrgxLsH#nm&Nde2gK_%Z?=gDwsPF->vi@-tP zATSQz#7j0Xi?DfNy@kd(7V8ZjMeZhgA~2fyH@^lRl1*VN%Uw`7ym6nb_0!fH_+)-4 zqHu?-8Bx$@6ZzcOSdv@lqH*gs9s{aOEcK3eGFsBg zM9e~2HI3LLcBl8@vD5{@;Icu0IMziI;IGXzKfdk8f1=4o^gF!aIv+*mwMwKGH{_Ry ztVmL*9*k|OSz6SMlJ;ZWwoy5~PYcLH3)J5*rFH!wtc4Bcz~c3gE?PzEvvpNspDUh= zl)(--(do*qX%GSXgGbhm*$BA?#@v3Jph-+_2c6t#v@e)|#IedgYH;QKd3hP3v=!SjiUS6VsAV$y4WP*B>^)u@aA3bm3xy-&ya5!?kC3A!j9K#% zNNBa&MNO_}Q<;qvdj(7u*R08nUf-78%>jYN6Ia!K{E?@1eVD*qrZ?OUuM;)&j#i2z z0H`=u!g|1amne72&DSNaB}r@Vsd^;5i8Qpbn-W7XqsHD~6N^CE#Yp16PY$Q&6K!}6 zYL-d0-1IuQdB;?(xvLhqZYOlxPmq8Q&m@@T%-*$-X_xZIBU76GzBWv6?^Az(aw7!s z+bN9J0r^8{;p9MXu%TJQZgWwSQ{36u>Z@+`Ze5^A9wKaEU}t0nS=sYSDB#}r{7J}qVk7yN;_ z++?)o7jsg%VQYE0F56O=WzsK8u!-?u3GhoY!w%=4U##7j_yyb=<%@jZq+178B(HUl zxyk9ljD|w{d5Jf?BGHV%+;X%Mv~?+ifJMl(L2KthzSwy_vxKyz2O!q0Fi|Efwu)fG5550%Psv;0vBmrc@%nW320d+;DvXM-Q zdM*_+K-M5OHx75e8mo|N_6@!;c1Y%$p-|htU13?H^KQd%1$Q1qPIcqYK6(-*@?{_Z z4BHBJjv}boF;&D~F-vrUyI&{^U$kN(hWJl*u{!J-{z53!fj{8?>T5-&b*7G)!GJ{^ zkn@VQ$>*d4i9(ZV1JLm>|A6z%PVZ4e@3cg=SV zhFi!jLbBz%=@HXk`k9N>ASXqQCd{x z#Me#M4C&PBC07@D*^hN>QmJFL(7XGFbO%lhlih)FGwJ=WkcO<8UF2tGp{~1We0Lfn zNa&F(ZzwD_KsqU_j23)#9KY=i`v{j7mC{Y`xk6L1KZsbxI^RRqp7`kxbD}+hw2i9= zg^h5wqP~Cv*ZXDN4J}((kW2!^eZGiaqx9|_XRisN^Q06Wamcv}W>vaH-(CD@RT{w} zIJ{Z>KCkW2fX_*R4rD>}t2dg4vhGwqFzwtBf)tn!N;UTo7Fbn^_vG9g%j z(UV(L1lYRUVRA9I{D%Co@_u@8ykGFJQ;^O~w|FQ@M!3i@JsQ8)M#@=@NnhD?k5-FK z|320grtnaSrr9~iJMGDy3{SV8GBvvGNi^$7ei8R{>};tP=N>$ra%e+ZS`G`^BQ8ld zmTc0E-(YyA>yKpaxeBOYtpc5E<_DZkQXu7Bn&s@qBhy*x%trf<`{1)Tdi!)+h0z%+ z)2$W>`ol3qKpxEwm4b$O=Gd{I`)drC+y-sw_;`MSSc zTP?#mfNjoa`Ye9By210f3wU&=_O4TY>A#s!_MTh%bcl(fK4;WOqBW zF?Fd6LMknTEzloXrhOLv$#l>nT&!Q&)VYsXx!m&Mwe-@l$$-w3Os;JmE<)(&D*8KX zeYyfUc_63?uW_wwd1J3w%|%$6rll*vYa#AUL0zPmv1sUiD01LN$kwvWudi>_8>5b( z{FG7RL^zZZ_A-QUTZ zz?7Qbv7DP?J=Yosm#aMLjnvc>wX}l?n1y`J>e3SBq4_IL7d;cF)42SGXZETj5?u<~4TtJWh+I zY(&Sr>gg^HXeQ(j8Lp^A%in!xPEQ#$bMC=^jd3Z1dZTE4yl#R-NKvPXCPZr}(k{QJ(B-2Vll+ZrvVQco$4=VB!e|?G+S9u;b zQa_NJKK}tk@!VFk=V^_8J##-kHukb*Lc%`bg>yXIYJC)crr_>~np#FUOc9H6ok=;s zdzkBUiZrgBA{5{{+lv7vk!3mz*wILWwDbL&Ifv1RaTlo7+Ln7Vxz0ZW9BI>_S z?J31RH-=u^;#Rq4_N63*`%$4)38@>&+&9Zjzau31YdFHH$jYhVrswxM{k8rMtf24n zD~dBRU~7lZtaKR0Ov1HTT@;OIXk|<&!#Jt2oJ-t&3v>r5 z3{=t>0xYUBGZJAnw*c8S?g6j8EODdep{{CPk3CoUXf)d9A9b%QN?}7*eSdj&e0X+r z6rCfJ7`{1P`5{#g&5mZyskzA1vZlWM6I%L_+3w*Dhtd*sV6CMKQj|z&Ju^_7j5e0V zyRuSsfW0jfqlszNdlRl%>0Y*oJ4n&h+BeUW6K;L1mP-v?*2)dnnO%m?y3A9;$9P-T zXe(aoCJVu?%}{IHzNYA`7%63P#dvc)bUBJ(8zekbvTn1!!wzrE4OzNSgSGTVmfhNG zn5B7aW%<=M08B7=Xkm$9)ab1*5F2NhhuDz>+{nj94?7PTm*6`sC5JD|QrJApU_czi z>05+A>rcUgPW7ritht3?c6NF$^NFcgkKwFnG3Lx-tWz#OWG!cUjyT!DJcw)ur#oKQ zztvwQW*J0K#l|ZPB-9d%u)ykQ)=?UNjIxl#uB7=HCzv7O4Rh7q_=IE^)#9<+6gKSY z;smPYa@5D%k@0PEbcn&`(g|Y&O5soT?(Ssum8fcsCu1z}$Bc)eOO*Cs+aqU8$bF&=THsurT-j2T<#XLFdCvZC*oT$mQ@wowYL5v7}-K4 z5^D6RxwdS6Eezalxibnvxh%9cKu9`R`SC4n8;;#AAt>#Pis5d8Pr_(I`#H^%Q*mWG zKY=>wjO9&7foirdLb_J0{RHC1n>{CDo5MwlZ=LUn+ou6D?0m2jl`*yy3S0hkQ5+w6 z=bg=e&1`mBq9#?CUjqx&v&t9Q5bq4jFwBY5GlfwCJdEk+VB6^$Au>fl2{~9O zS_f>7F3XZ0lk~BE@fZE1Io&bDUo5Y0Fr#GG=H7v_=6EO#_LWqD1XM?0oNzKOhqxbHSCDq@9$I&)6Ij}gV zS+Q+U$_mMk zNPsw_w4pD^YmC787NtAf92bi*7P59Gr7tm-v0l23%4WW+wOOH{$3!^c;#OH==7G)! zDx4kquO+|Q8BP9kRnbH)IT8^&BT&lB8{>N+pjs=}q8e>57U1>{GF6F=u@S#*#PKiQ}+@cR_fB%Y{n8!xiS0wI{}8GcPXKC`*I=!P{L5Lv|Fl^k!X+1 zx)#i`Q!opkE5D7I#&fLN!e#Mo@ zTEQVAaRp3CiCrw+rL~Hb&Ws45M&W#z!`Kh2C68aZrFpDns7{+HqjE8j?s%@ktV^~A zXx=?ktRsm`@rHyrcX6Yg|M6vIc5kU|v89A{E07^F1>L_z9Mb^$o)|hEt7gG>Gjpy-%C<5`Dk9{YWxZjUPB;1K3L(P09krg|FGo(N z4Z))b7Jz6d!7h4XQU0_Lvn3!73=NE`FPg7(uRt#8&+Wgc*6NLKdsI~X+9*}xmz1Z4 zM$2st9xfTs(p!^j$*GloS4OYP(&E`}T-#m#WiPJn>C0x#nlv$}SHy9#dQM(NwUbxz zDYGa#dtHfN(dZXlt}V3H(_azGl?@xrOXSNjio9O;k(BiA(e(5SLLwNmOMxYf41B~T za_Z)TZGG$6-2@kH_Jmj0y+3)=zZnci-(%}EdwYBL?$>)yo<4j2^9YL|E}OK)wlHfO zvLVfVj1`r}v=FSC&_bi^h0y#E*S*H7kkRWTy=3N=<*{*}j~+y~1no3#U#R?bZ}X^U zqPqNzi}ae!=MGvcXv6_K?ytb>CdR5<91JGc5x0|^p1m#L_u)TGfTCo6+gcbv{6b*Sa zhqWyz4pCJbF*y~C8>Y?>QQS1o-_O|7QbQ(TSW_6wB8i9PV6Fj#wpEF9j5rJevj|RU zX9zes6i5Ysi#bv;qD5-pc;%>oi89LCWdS?iiv)Dd{OkX*NG#}IY85iqv$lz;P z2V1Y4H`qxZ)|bOlC`&Z4e?p6kWtF@AJ~yGF9>}x@VP{r^6thaj7GyT*$E-@O59SIx zTupVBZ)*3&?j!5*QLDr}|8s$m!6jD)bN%+kv~%N3^V7G3M#Da3LtF%!OBeV-lVvUq z%(0kqDBFL4{1%UySoF&|a1UlS>v*w~lB?s7kS2G5v_k_3~HZ%8)&$8QhM5tgiRsh=rNt3S#W0 zDCQ*9+0w*Hk#A|nv0O{-pprYb!Ib8gD?>qA=JK8D{AVznU0pl-SpiR4oM8zw*7dmz zSw$??l3c8tOG?fFFZ(lL!i2#4v?!}v6zh9^iQLnfYfBS%NC*%uC4KhHZ}8v%Xq;t(E2*m^ENYd*^b6c43xJ7`U}L%PB6cD$D7) z;Qj;5qWkTcOq9W#bPMz~s;d9lYkmSKw$BU;$=7yhRyjV>0%NKkg+RQ7Ln-i@+FzVF zEAp92`u(}W9JLOF8}%fQcq``Rbv3;6g0;|UfCNG};bz_5z8~53q zNCh!hb^O(W@+FL1iUD%pyYGei9>*IEZ3R^KW2s)*6|88&noip&xXf;Ccz>@$7pfuB zJ2P*p0sV^=BT8y0h}>U*j(Tah5{m+)IXsV$mUHuV4cn_( z)g0|HLv0V{h$m>E*5fVoAw!bG!e5eu6oxWThnr06+f(UtPGy)dt%b`nw%+QROJ?YW}ce~+Kqc9)KP9>Zwl|8H5z!t?&6P0lu*;=q!Y=$D>B^#r_-jIz*j=Q&Y}>x-<^3l<3^0#2zS$nwzsx9 zH-xl1W<-O2fT#sJGI6KVNurKy>(AU4Kr_696deRO@TY4(?q8lXTr!G)8-;^C0)EeW zgW)FCvq!mW1Ovqvm`chc@vSuQ^4EG}G1*3)X8oL7uin=MUyyPoS# zSWxr&4CBID{p=gBkj&>wZmx}eISqv9~ivm47Ew6D4(fH9b7oL zU@%%a^k*aM>Ry)ru-fo8r}ru%nCGX>=h8F?cwMavS!Ddu#lrfBU`e)LNP#+W_xB#H z&7Ve94A?fd>_{{Vc4&%JyFLn*B|YJ2#!|;3H*ey$>Q*t+(7AiXt_d)XIl2@1E%2p# zblRqeF-ShDzDQ0^&XZc{MJ2H?S{&l9u-v-2vGe+yn;FMEGO*szBe8u=ZXh02($^xm zVOP82(mQ?h3CertdID#s9cF;j!DpN>)0^>MFt;%WcZX^Z?UW7?RkttwK8+d=B`<{> zKcGGH8FNkPENM#XE#xF)0zU-uG?bBq9|dkaSUTYqBZ!5^C}H`84coqUYc$70!?bi_ zUNCj#lo=CRf*_2v3POLFsNg-1vBV8t6+K8_oPQZPEU$lH=Ss`JxtcE|31~y4;12!P zXhiL%a3^9$uu+AGl$^`Jo%U+!Qmo{_RIbnL%OX0gNRKeii1|z@ET^QI-Cv@R4;Z2k zlNqKqPUtm1gykK&nGL9n9sXGBWKa*PYu?T{IJt!7(9gFYR+@YUe4pNRS6=TRM=|GL zoL2g_H=gSJP5novSYE5^B%U;BEf1oNYbZ>2djCCw6eNfijk8nSI2y0Q{lv@8T@qYL z@Xp*ep`XJtafBy(lPBAneuQSQbr;rPcO~ARXZ4AZiiqvl);dndg}n|E_D3~(<*7h0 zLFyIfbQ0J7nJ4kyT^meTKRu~BslE2P?j8;fII#jrW&OC9gM&nH_16bY{lQyUk0I^2 zVo$`X7iyc8dek^B@958NbpEnj-Lb#BkuIt;NPDmgd-u}B1RIQ66%&nG)t;VU%zK!W zOD9ovud$;)6jrJ?DU84F8Nfq|e0kPrMzz-`JNmO5HD8yi(Rt$_syEA~pNY8lE2@<- zx>XZV38*-GaMdN;d!%}w*u(3?W~6XHbxP9DUetWClQegTFYry}*uCuLUVu;EcJpw& zWN^Dl23IbXUsjSTb$-{dD)1(%zo^ixxdfV5gfPq`3iw#B{2}ft;D@3jPH$00vd7)*aW8v(l081P$1|+tLq7NHGP-A% z(>=Sa?)hm{Z35Dq%KLX#ZxCf)%hkSHNykEMPau7m&WW0uDINKF1;Q zq>+46Jv%NnE0G%=*R)cZM%w7A%}Imr1?Y(*Qm#9SYBMXhX_I8M;<`eaz#t3BjYm>rQc<%$AK zY?TFg3=?nhcOwk2mj4|m?*~>^U|wN$3c?~lF-P3Xxq1LXA<(wX^Kvkh7ep~6EN8zX z{*UF`j&8NdBKrrax7BGAR0P2w0T;G`T_*7ZwI@G@H^?#WP7*Ebatw60e6WLG-VQSE z{=kd{%>FT+nzXP(Fav8{hR{Dox5X_JttmUvn=K0pF!`nZAaea`sR~P748zTps@qrm zVL`D>ijqK(uV3n~6|f8qP6jfwCFPB?yY!a#ejYx%J##sHkntwH3-d|xcOxzbo&yRI zO}r0ur>EO?<+zZ$1oC}-l4YXQNXskC^+s3%@bNlW)1Midd^eEKkU(YWjD@b*U#J+V z25khE&IzAY(%;MuxGvr)z9jRT*@-vpyaiF1OBsxrE$SveC&e&d5hQ)^u;u;Cue$G<9hdnM zB*aXRhMxnHSloVjM}fB62NO;)-w#di=`z+mh@k=6_AA(F#VhMDy%Ej8p38Q)f$uCM zMQt2`&p~^w>e@ZFzf<^rRoIpkcgg$W$2=zq`{n$t*J3kAs=|_8PGsa#X@Y;5w#gj3 zxQ^K2W{$#ED(D*?Kzo^G{f!Vb{>oT!)1}8!>A6{YpA%dp)qAs-D92eNS zPArOvsWis9N186L6Cwd= zo$;+42+oUEo&`qRGE|wxWoi;S$3Jygf|!C)fh^X(<({3OLACh6Lp+9@r;MWxD=_C# z0jX;=Too;ICqSC=I@m&0fm6|Uye09>c@SYtZ$}*@E$j!=Qk^%ERj_^A=X6t!o+ZS* z&An&OH_PRdPd87`HYM;`;_erFOA_}!UrhAb3)%dO1J7!CQ9jAQvKPS87b~q<&%ny* zi%JVG26l9?yZ3nxGXE~%+`GXalyku8iy*^`z$)c~3@m#AOfPH+E62x6u$F$9$6WS7 zHRbS}ez3-Ynl!-|kxOt{vjk-k&I(zI31lfGSS{eMkVOd15>!h*!Lm3kW#!KMUsU2wL+n=`%3l*~;-`9E!@lslcE`IBcaq`kNTkZzv)w)3_2tRuPYYk%yS`AahvDf> zK6mcRo^l)X-cw-oL>qh>Po&*(AII(&(Wig*^3VQ{pMLN=AN-h5k=@s9e%Pm%2AwCH zIKVWKNBiAFdZ}L34k}LV3B;~(aRZ8<0FagQVX>UO`;=oq`Eq{ro6$ArH>xOliH2&E zr_Eho$&Y?3eB9f6ntj%b>&=gTI|X?5JO{vw2Y3(wz-tDe7Z3200KkxB0DAENzY+C1 zn>U?j(PzK=;~)Jky|CHbjsE!0UVhO1@rOV9(5I+TA?<^ZMms{J47C~9>lnva5j>}+aMMEL&C*>@&= z_%2?P`~IoEi-i4mJ?i^sgd6!brJR5K)sH{;K+V#_Z$_W}OF#bckACX|o>G!_G#Nhp zB>Uh9I(+zP_QA~o*VPXweLQJZxIO2cyARCj`jKI>+pf*nO}3%a$Q-p;qbp^(lSQBX zYaf2_JHleOefTnRiMF||lz9(je~9tKGf_Mms$3PkIWj*ve3rugM#S)L?yiA0S`9`2 zuD}`l_3)b+$$s#`Pb|x`=kLps67dIgky=3Hd$MF?{lN!6RTfbEo-Bd%e&z=E!w;fv zyEAER-hwU7#a)nk81NK&{*S-<@W=Cj>R}Po_gTO{nkyAznFn^g74VPefYqyF$-bWg z{%{Uh{V4(~C5#r1A`P^QrOg_i9^AEC8$wp^q7t|6`p3`y+6TY!8^1~Kps6>bO(!So z=sjv4H{#dbi@4z;FenY(ghCH+N*MWkHp|K9Vh0rRm0b{q<@t;gC?9%-24S|)MiGbu zFW~Uv%QtQV9Cj?dKE@M@fTijyn@WgtDL%t*`HpCh)0nnyxU2OlYVs1NZyxfuJ>*%r zQ46)|Ip2g2Kls=G=Z66|_;k*=5~DTPbeH}{xc)WHPva8NXj+Qzb=2#!gD7#j5}!KD zoOt7?bowYd|JmPEW$o@JzSzBXnV{dKzW%2FDia#rnn!e6qveh8F7x;8`b0M_ai@&A} zpug_xm0e6LEvsg4VFwGC5fEiLFwM!x!)UGoU{g|+&BoT?22KrfEp7;@pZZGdz6AR$ z%d-J-SAlpTh|fHPS3gY5(;CaC3Ev)v+^Ki+FE}%GRFZit6Wuo?o_}bIA+5o4so!Oy z>+ku8TfU&qAS(d7BiaAYdK3KPjSdP0m3)@bq&q`<^zJ6ArBA>8?|$<4|H1$MxZ`Ha ztk-ct8mpZs{qip;duPZn4WKk)gF@cXp}nJ72npvOZu)9wgC0=<^F(V2`3Zo0uyraa zVXFeuf75-p^$W4{O>XvbET6xLNdayxqjj754+E!IOMUJ9;J8|jUsb-dWcTr)@`76m zR(@9AHTmX0{piQP@tZ&S>EHO9zxT6$@!_M-wx9jWUzJ}}&l=ygx>s**zW-PMjlchI z{rmsHfAk;!!~gU@`_KQ2|MI{7NB`ab_&@zG|Ky+kZ~vDR7n?|71)ap{Fh-h7wUxn< zwWuEh$fBpDey$6jIp83m=j!j(Z`OYEVd+-Z4wvq>ZdMmTP(2ocmD%0rg-V1Eq?ZCd zkUe7Qz=@`1vAkvPCygR;l_6hr%+6m(MtuYzhp@2@F>{Uw1o3-gSuAMSf?^Ej(3$op z-ZtR6wXJg%YDblm*YF;}aHuG6N2utyp9HY%dN*E({?jxY=d3>q!_a83~KV zcs3f7fjdZav(+v^nAnl7;_v!IorE$$OnIEA`?6eTIe8j{OT}$_C1Whdd8B3a;c^#` zP?mIa*@VeAmBkpBhfAAn>!w4#jrXBap*KgR`Rw`XMdoI=uHcW^u>)=z{n;b5(fjdg zgm@ZTz=-W>hwhBJljuBg6)m(asM?6GIdoYu2R`$0KfP)y=#bFM(i zeXPQ;`NQyz3LYR8IRmox&_dKApb>(e)4UMwGG>SW*v%h{QT&j|yuA)LF+Rk$Y(8^u z_t{f#c8e^ULVx7hkk? zKYzOEB0GlHv`zA^4xJKyd);pL^PRojT^~OjR+A;UC_*GX-8tvqVf4@c-%lg^OpH|@z6iK8B%C$r>Y?(H&r z$5j6!$-TF~@7XIIn#{G8c4PLWu0jP+HUv9p4cwrrWux0M?HAs4wEl2s@7wS%8gPdn zN+084xk>57L zwK~Se2h7JvC-iT{oFQ9&4_@!>L}vDO@Vbss=&QpT(gC2$% zq_}696y-=mbVgSu*qBGSv_u4Dj@pw}^!#6O|3^E!u=elyHyP!!ISH{qWX@orkR;Yi z0?uI3NDE(|gEM}n=4*S45|)?IXmJFIu{6Op0b@D<@=p>1Jk=k`IK4EnFMk*GYmfn8>m?@Y3WwZw; zyyUB=(+z(qQlVKc-Sq1-f@GT4j>LTvucO*oqxx-BuP{^4saTvY+)bon(OowevtGY5 z*+#D3ZjXl(T-b;G&C?g#TlQhwc~i^RcC*{RGz%$kL$kO2k9x2T_LOuZ^Iz!kV7``} zT^9@S^DjyVWqBKC2`vks7!j8tg1twlrBiWu#CyW*ODBEWI{m9{Was~(?mfVwxWcyK z*#ay@ny8=%h^W|6QBhY|7DQB3RP2S0AXS>EDA+ZY*n2Orx2Ulzme_kq>|*RuBbG#s z=>GR}W@dMGHHq*0U)T5j|G>TXInQ}cnKLtI&YYT^PS3OccwV2=az32a8+c%0rF8=9 z9sl1OWQ7!lg8f>sUpo?TZtPdhe)I5a%a!2QPF766ql)lral8<}ikJA|WS{AmD`mf= z&T;na_y5+9dm&<$=_t(>I!cy%u@1ihoLwQu<#UQ+4(B7oKMZ-CqKM=CI68e-&WO$%;5GgX8e)!EqW6N`?603X14gSBxK7 zF{MOSL|LOK^ds}PWDw%U!T&0m`QLxWH|Ml#fbj{Oc5YL{eNyZXa{w>*Q5)n(J8{}u zz)Y2uc5X*bs{!Z3u4&I{H^BS^yd{j&hQhq%GN)bGj?-qtd?lRImXIFY`oKO3Sc-M* z9>Dz?%wj4_V2*>#8DVFGIZK1v6Sx&ZR*EpDg5OL0iT>$4$lM~pT0Kx>458g!eM&05-^ctUBdJ`Y?5-Z!^aUjzFcuvXs!KLCCQ&O-hY z{3q}?pfIymhXQ4Q@_-Xi3Gf7}19bsEAP{H`bOd?;-2e+921o>QfGj{v@5~9vCqQXs zrL_T^09U{Zr~~){Y9JU014Mt^cbwJ_I00E7AP-2Ki~Iwx0Y3n|=5bmTPz*iMU*SSd zt6s!ui^f9-h=4h78W{ebBONwCn}oU@&}CA$J?#&z&%7}HB}4Tg+IfNRc&?Y2oth$s z4W_Zp^-W|FXO8rx_W#t6#lFvygXx>dB+gvD?7VELo#|teLsFVlYB1LuU=%#6@BDiH zc<0w2fadS4SPV^)gb01qy$pwuzOfTis+WNdZ^^-yv8!()l(^UHj~x^w2d{4;l(^R$ zknxr92V}$>Nev$ACB>#2Y#hgjpH~-R>1pXz>Sg5`g~aPaSUb0n=CK*;-5!qq{k2t)beSUynj2S>A`nwJayYsAKn*! zu5IMBZ-Fb;R@(Ofy^rE-P&NYIKqxQ?@1bD4N9cVNyqVLk+=Q|pe!@4P90b(s@eYLF zmEe8gZ1~9sOe4GXBi?&Xd>qbYc=spZ5-P6{`vl{&g$W6XV>n-y1TmeE9mFSIjfyH-Qy#Vb)2&onQlJT z`Ght%x2;}taLwwP0*nH}Pqlzv>c{F{1^%wCDwV6Ne}R){fu~bsBt9I>Eem7?T*8D| z1ET1dxm>q=rb;1~eKN4(#Kc@ox zdDSYYDlVo8I%UH-m4yOz8_z(F3-om5YPl9rVE!rp{` zqJRr52;@98fdR4t3aUVZqd@K|FDO7>6p&wk@_|4*Rq0n?=~6SKIznlvkQK-j0niO_ z_t1E_2Wb2>0kKXhr&xtDPpOFR*`|FEM{&w2M>!Y@lyaF|Nug^vg@uaqh=|D36nF;* z`}qY22YchkyTC6ZBEp$d4Y!m#uUxq>uRx_#=5fKpG)SMOC0AhY9gOTmHsvFZynJ#c zzka38US7_GQj)1c5sH^m84&EH)@VHPU6DRlwZ=!I4)#U}VN<70%@2-?2n*&^O>lX* zK|o*=6-UXFtE*S7u6A3tEZVU-{0>2ay7sKnwd>$OPb61cr7P0x5z%$9#>TC4rem|N zUAy*dhS+*G>lzW!gX3D|2XgXO!?c{NRla5zF2Jx7?`qy9`4L@P(fQ^_ zHyeKZ>f4{#u&2Jgx9{H^H{Lpc@>vbINDl+LfoxOdD zeF?0kkiCO5{&h}HZR63XYTdc10$Yppi5T*ESD znPEO{+qd`e3C#=j*%#9*q@_PDBR3pFUx(MCBRDuRI9SuEZLmfY+_sYjk;xRrg@{+B zN={Zaactt%#N5u(4uyxaGm>I&&-rL{806wEleuv&l`A-MY8;eG{h|Dc(>ejB<*Sne zUylA;`T8FIocH0IV=wB*CgrOqr>la`CWVEadq4qKSX5l{zV!1=&B`C@ZT8@now&?t zCU3f0xCRB3J}JylY-E+0SlM;gYbcDZ7ADsBdPOqWkriK#!|C&mD-f;t5{6axQg9Wl zI>AaI+HnP+>Dv%~d27Jug7*QhKAtcA{AORvm&NX%3v>k^b$361)-mfboz!Y~|M=&; zp-1kxm5?iEp&M7qTi>f;mFe)!JM>7ATS*DCz9}dU_;}>}XWpu~A5d8E@yK1lYVlaj z@qU@Ih3O-aSNy$;v!%lw zw^RYUv+1YUl`CM@&)ZYSPj)VbJ8zw}aoCPQup)z?_IcIDMU!1%HRr86Hon*~99A5% zLPqqXFc>Y0NWcz>(XL34A)4Z^83PQ`0K)|@5RLBd^+sGLQ5vAr8{Ksu#SrXtA9O{0 z9AuAItN_dnxMQN5!SIDHY&jF&5)4AD+`Bm`Qy_C75u|g+?z}el^6$?^#7cmMh zKrQ;>hUW{Bq1oYczMzVtLktT>+=={QFv@foSF}8(D1kiVHpp5%b3UH*c08 zT>qixi;zn3z#S?aMZ}*HZ#q zPzQZ-l+!Lf%xM#kq1-%#G7#psU};@f^ymlu(vzGv<22t0g8dQ0Kb3>)~8|{45Rfr5_7yGv5Ep0nVFg$j1^O36SX;6Tn@&@gwTri z5qTYHN7hIZ=$#`w_b`>9PGO=GgsD{~qRtQ*4ku@sRs)olmuc-Q$h70ZB|tu~9hk}N zsB?p<-rqkXDV?>vV~kP?$83YJBw-GAw@RaKF}adk4dC=Cz{^Z@z*c|ble7gz}F0`>vd zfLp+OpcHT($VIsVet;V20rUa#fSC03?3DCq>=Y2mHb%hi^YO7cxQrNOi7$G#iYf-} za}qra;=-uHNb2>#mP%hb&}s`@7Diw40rE?MvD4-ShD;M;%*1XQ?D0fAlsmkbI#u+W z0mY*C&Jd=-gvDm~{TCw#({vAs1OJ53gMq#_>^=fLYNo()C^#v${6r)cyOMYip4b0@ z50*mesAH*4rBkMdAt{}6bN8rR$+cP)&#GQY85oR!h0d7{S;!aaEmY){$fD$|=#xi9 zvh85`B{rA}cAXTMwi#i*DVRvCqAiRgK(U2#fJQ`6rs*B2G^1`mv|Pr}kOY*0dN&ko z#>y-)dS{IIh(}2k86AURHJpWvlgTpcS*wQvqk+l5Y+wnn2G|Ph0geD?fUCeA;4x4F zd;sM2t<~0m6Hpnb4%7$yfIuJ?$OI+;}#VxB$KfyaOy7SgYNE+CU3{bUFwu>c_&K4NMT^3&HCJGwJMw`7m%2xB&bL+yNc} z&jF`6ry7*~v@j8WWxu?(%$<5H*-Fp2BWXpXICddeN2XpZ1h6Gurk#}8RD&I@J9ftX zL6wT>jrcx;an-HW`0C;oz%B=`17p4zcL0p(bKDiM8TcvK9Q+ZC3&}b7TB}vyDqzfD z;hKZ3z#YKW-~_M@cnG)*_+!D+=a$GO444e#>gGYlE;N@T?_yE`pd;`ptmVwWL<={JD1^7K!33h4*f8eTMb8sk_ z^S~!7SO(4l%fXXK?t}D^9K4<6;8S1)_&!((E(M!`ZP0IJ4t4`ufE$2S;2^LiI09@1 zjt5(V2Z3$Clfh-c%fYtb-C#TLC9os-8MrL?6Bzx@T)7s=cW_;Bd2lee0=O^O8Jq=n z0Z#{41aARX0-poBg71SHfjPCc8Y5h|3gE`zy5Kl)AUGb}6`TN01@{LRf)l~B!AanC z-~r$x;AHSka0<8toC=n=v{ugnmj}-U*8xD0qd*cN;hYzKY~wg=my zf71b61MCP60+$8%13Q6pz~#U*!R5hg!4<&!!Or0GU>ERXa7FMtFc*&JUxV`kdxPcR zHedxf5v&AH2AhGmgU!J=!4}}RU=_GRAkGWy54HmL0b7GdfNj7_z-7RPz_#E=U@i*z zuSLAza$q^QHdp};1}niaU^DOtusL`sn0nB6fo0%RU^(~!SONY3R)QUZ;166IYz}S< zwgAU~Rp4AOSIBX*!1zez)`8{VOJHg!cn4O3s|F(+a9glBI2~*Oo(@)lH-PQIr@;>3 z-@%SxyH>a^;HF?Fa16K{cpA7ocqh05xESmVmWANDfIYz#!6D#E;AF5X_p$QN*RFjvHJ&A>8nI9Lu&2P?r-!Dis? zVD19PT?fm+e}Ls+$F_(M+!(9`_XV4QM}p14Yrq!ZV_!K%#LP&a7?k+PB+QR8mh)e}yaQ&NVFa{r zo@|a922f*}UiJ}(*=jUAEjk7Rt78NU?}`k_<(OYumyS1KHVwc%G$?Hw)@`L&GFj=&djm z+eL96sAEZF#SwUq^|04VuE94@hab}XvG4hz`D0L;;uaGUfC1vh8Bm-NrB zcvoY1VLTUyeN6bvMa!2bF&pM#OVrurLE5HrQ7XU&um_v~XTTNk0K5Qipbp>z_yT@_ z8qfg2KqwFfgaZ*k4}edh;Wq`c3?LiG1M-1FU^Flum;_7(W&(48g}_o^C9nqA0Bixa z1G|8IKoM{hsEq=)4tD~L;1@Z2?mAZq(K>LiP>ub-mBqK26K5%Ng-oZbfyVy&KvN(< zRvEINonM#EbrjXMUqUy8A?b6!+Y3^yeY*X9n}aZ}g{;_%nxj={Nc-l=!oNdF40yqYEgW zALY@SZ}o@F7yVhn--d7XhfEaxS;60yZ}o@7iTK(wXOW0I#@eUMowf7-_5``0KtxF8<;Rat{AbJLhaI6x`Nq_ zTK}1aUT=|@`x?p5C%|lBG-8SYGmZE-)a9_Rqp_4~#X^4vhnaV|nDxolu0>sp^;)1c z7hM+dn5;p8XQk(4Qr`zW@lDkUm=KblIheMpmq=klIxrAFmE)J{rRiEZZ0iohXCf22 zEGVs6DcF+?HNO}(y^CsgnB&8tBPTT{kK&KTgf70XB{C%g?Wt&8qgGZyL1@{Py5K_M zpdJcp!o)dI?H~*D)Tk4S#MC^8nmks#&Wg?yRu≷x^7FI%UcR3XkqLI_RFGL$vd@ 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%0DP+A;b81N@9EePsk-uYiI&kE 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

AOHUP{o*fRv~Cd6SN4qBLKczQ~18jMH7B*O| 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�^0xBfI3Fv>?00`5tR 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νoK>^ZCdccnlY$kw|K0708fqyvzeRgwf{I- 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 z zU}u0%O~(ri+a;^n@Kw^3QAP2-gF=*}81EL*{}y6Cr@r;@2&WW%DTA6_yVpl)dB1oA z-~RO0@4C&k1pAu*C8)fxb5GxKY*t}yWEfjka=Vp~1ujGAL5o4vz!W}s)BZ(MTj-y5 z1QmNvj5nx0Qx{RIFR;pam^q*tj;Hb>Env{JehcF67UQn`yWkvKl2Oj=sV_V#D+=pJVg6Fn9>DHojG!M25$v>!_9IsQsORbX{RmF7UWGpIp%q-wuQ zvFGSvB{l=j@Cv3V)EN{V?u+~jUtQ$2WMXtqpk+==K=i|c4MPhZ&vQzk#XVEh&-nI1 zpl~gj{xaYnyX|dpmm8dk_9A=X5qAG!74V@k3!@|9=lDts`8f+yFA0r7b=KP+>+MtG z+WVLgd%nr^NVcgpyQIA;bXGN=MwuH(a(rfFlNi5;*dDx%WEH%LQpU?sd7-i>RaAfH3Lr^Ug%uLb zMt10Y06S3cCcf3U{+5t_RqhHHIrVa3MS9-x;3p+k`T2nBGFUEw(K{rh?-AOI=Tk-7 zNoNbk1$?+TVa!kFLJ9I4Soc-$XIXwcC2kX6A3z)*wwCspN?$UUtYpJ$i*|-t$;;!( z0ewwCy?YiPLh)c+ZtVAzd}G*w8Bc;Az<<^U=L1pg^PinTOYK0|67{p9NC(R8pi6AH zdbLo_cuw>=L+eiTLEo}-|6z&7+1_?5TFtebt}@d2enY3puJv^8w-NsPzN*k7i1&L> z_Q1I^+&|2I6z*$ZzVRn#&^In&Z8=<@SNr_qL*>8rb&-^uzuv{|9LB|3#vk4n@7?jW z-RBN^KI&_f7J<@ign8}Bgo|-;N-%erj+p<|(vA$u)JC0Ap?YF`sV>Z4Wn)f%{wtvUIv8i-p z_80rXI3VPU{huzS%dIY_xc>SzuIf%h9TvriboB7`icqYWEY4A)Z+V*CVRoJ9u3(zVLrV8Hv6=!}kwH1UN0SzqV6 z5c`W`lAtNJh9MTyx=lR)+?$|q0$iTW!p)=&^b6xlh@;t5l*SwSW?B#c*Hw!0=zXP} zn?g#KuWTTz3}(it8F}t?>z)oyQWlQYZA@gNj)zTEvWkrjk&v&B`rD{w_$VAZ7TH+o zYQ>+ofPszSxe~&k`yMGRQx@*LpUI<$#?5ugRvEamT1ekD-m zPV2ZIuw{_M{ahN6xdUbM_q+W&=8mWk@<&Mj!rm!4n|nyImMZ*T?fi5UF5FX&jyU#A zD8b;0wq_8^O*2Yj$g3UzPRa?XE6J`Me$v7 zm8RO^&Z*vCSTWTlSHq}8VMT;PEx`ib?z<+ZW{y7NXGf`}3#xs0^@`%R*J1a$pswo^ z!68@Xh6f;krdYv{?R6O5%5^)`6{9xacFj?F1E5>>Iaa=V&(a#@_Bw3EGj!~@9qOe^ zU31i$jjh3C9?V*Tk~u0=%(9sM6>9NI3DatUWSM$@#prq^(X^dM-FI~gJ|Py&ub}d z2S4O_Wmc6sr0VU}m{A%f`VWM9sftFTVNapdUq`bqNX-} z-BCLPBP;(XVCV#gWG%ygu^+hg$Jf;a!{{b?T@8Qc&E& zLZD4?Hf#{uS)r|Z>4dz9AAeSMbULZIJ<)7SS33m z>Ny$I__NmHml{;XyXhr#oTUC?Ywhp1?pBXJt_h*jg=|ByYR|yRk$di}8o8h%*Lu41 ze8`mZApUd^tQXMVAk=(dN+g_X&#?DTBFF>Pjjfi81Oo!rG}Yl zOtFX*RsI9?Ov7Y-A3ZPsH+rm~C*?oWb3~-)=)a){Wez@Z&hLjQdY)3`H2TViXj+;e5?c0e2&DzrwR4!v#r{f!dK9cT^zA6Nd$i5{F>Uv1GVHi! z6i8#|ZzvAjGe9ZhenVU*WVzNK(}tx=BLg{N+LA!)1wz~IN|C&${N(v+RBVE48GW%( ze>@Sa!LHv6Q^V15Gr};!1avA$`^l&@7JfAWl?}9^(vj)^^Yw>WoSQ=*(Ap!T3=?l9 zDpavD!-*>gv1#==Y}|= z|NnYi^Dd=)ukwe-HDm9e+h57TX6ysHfYVgF_t~=;oJsv@ZV1jBIJb~%U5AM}y-MLd z$+!YAKRs7CPKLQBj9^;6AYk6QB;bv)ST>SR$?V#cQ4QRr)Udq;fdYlbkn>L$onywcJ1$LRPSyZv#2!RTNMDC$dK)& zQ*!dF)U1wuXa4RnJ!}eU*B6~ zEMQ4>?t;Y)@^sA*Xu#~m=x+*Z)lurtH{ULO#q6}Km>IYz7gtZ;z||B??06}%IFb9p z_$f}yO5@ne!C;!6lG4HI63)ZU&0mn)ph*9a#zqD@FZ^KSU& zz9Q+82vCul+M6;eI^K~j3wwch9_{6$Y0Kq*(Oa&ZsjPUcx8;bMbW zHizzNX K8(lVd_O(nrN*Qx@9qz2D(h$VI`;Rcj9L;A9bYu3WTR(BFCQ$X@w4GH zHhSWE`4Rsn4R9gFG-&h}ebFPAU3cT(BH9XP!rJ7vNxfkO#51XvzfJM@2M{;fjzL*& z+`XGY`IM(QT1KaTHpGYb!YG2=?-Nj-R-r7w2?snmPM)4O6t<6ef_nLX(pcJC_>=UB zk&KY^+#!(`)LZ4Q(Fi*(AeprZb>$ zMj)M<#TiKvL^eHA8vr)Qv*xErAjO3H zhvx5v)Z0RPOO{-g8y24Rl5{Q@iBL}!L>>MVkoA%bLlGfaFJ<8(i+5MbCV6IcQ{LnU zI|B$id0(-Fo#&K0t4drgyOz}NT1tm?(z5G``dt~o1rgc*xs#_SUg^!e$ra$|gHTRK_NKK+ z@6<~NI(DT%)B1JH&JpBjL}h`(AZOJaSBkLyKE%Hx#5U!N7m6H)J;Un&@H77K19!Dp z!l(mWJc3 z5}C_RIzNJ4=Es+T1}ma~>5%0Vi+&UGW_5MUn72jS4KO5yT62BGI7ZEPjmw4xM{IWO zuj@X&OjQxOtLp9l8)q+WvR6g$uzPyPsoXrlVW5=OOJ8$H-PLcpYkTakL==uwLn%3{ zuAk!u53|JEB7Lo1>W0pb%4ZZ4yQ67ZHe$;Y;?@aPg?DF4+Rpiw$OjiyH-RE{7bw^t z$+S)&X&*5b`5WnPDA=)i55KWVVR^VJZ!(Ya&iPdEN?&0$IS(_hstY)YzGsCP>_ zJ?#0I5kZE0nmSEy_TZw{gV;y8y-nRQ&_EF}h0}8R$Q4w#*rD<>Qp;uA5uffWL43E0 zX+qbEnWRpm@~ECkrrH;fkz+;nKpQ1cz*ISO)RRUg`R$uy>SE_G-!d ziSl~cE5-;f92es(&rej*1tYQW`De+0& zb$ue@kYf@$mQXtvZ`m=7RH?Mu<@VxWtwD>gXi*8< z5%TOOpksSKWS8r3j{=~)qqOALL(cqLrDfYm=miH`}B4XQss91sUc{JIxjd@r;7g$D4%z% zt`Pr=aS_kS>OE^ITJ1B_;s(X3vW?J&z7W`)D7G@1h! zRvFDfM)Qb;b0CjuG{ZzZiF;_MM&g)r)4|~XJBTApY1tvAj2TMGSR+e10vV&VhZ^#$ zVLGfmns$v-bJWO$IzE>et+KdtLPyJ}`WD?OF@Wk~5Ye2!xR7YtT`fO7FReY%^icPP z64k{(>ih`!we>9zm44`R>`)O+e8(J(rh-*n^vKzr8@2{@1d^*KPM+*Q3>7q)xh`g& z9m=X^YK==vjEvQ29dBe(T+IB~*yY3SFB+KzM(YC{vyih+g8Vn`1flJi1BOxqhS0@7 z0|ZRETR0tX>_4C=}uWneB$KAbA0|V%LYbH;49! zc^MD!Dep7P-Yedc!uAYpiR^MprtSl51%*WsPc{5}%bu zrf`#Y-Q>pX%kpG!8Cq`4elA1Djj7@?IBv`VE+c{)Q_W>Wa%0})GW6V-CN3k2`$S@7 z@{HDB7@4Py)<=y@zR|kK$ZRlLQ;p0F(qxMa#c#2q zVbKkWHOM=`bu)}$LUX^>R8&^MA5J}C>#zk}F{3-2UGFS=B&eKkNNuxSHt$=Cy#sUx zPsyFqzT)@CY)!)UquVPEg{3UGaw>4GYA&>sk;^i#ruk;r=!-x3QGA^vh_lQW)7ka+ z#dLJV-jmxaw#WaiLV=YXtD3N5>EQ+Mua&ma*g8-;ZOIP>)K7%(w}QXp^DCv`l>vOBqrfK{JDlxZiB*WW}^QbOHO8%DEOS z^?EcFxzVjt&QFQ|+shtf+rs)V(&CnQo0+1roXD~DW2@K+NFZYDe~JEH+`~D7z-w-F zdsZtmW#fvSRrTRfKDzaj#@3H%qEuHtKqHk-DVw#*t01#%n`BuEf(0Z9CbwD%aga~>7^SRWC)oTKZps}oCy_PB($V}H-16IFoiHus1i-u7241=F(zz4Z_oKe292*}M{XjrDkU^;;3-~bs8W`73&Xl8DYgOF^ch3Ph|2?5*VHSk z^|vJs4ic}A`aQSfS@xyiI4G*;P+>WN_=;3C+!`A0?}96wc*7;GRN@L}9N|(|D&t6% zLakdWOmIA1Bk{+x-v9iK|G5^=oEF@e#+vRWESGY$xI7?Z7qBs160BzaF$npNQ&`XF zm=-tYtYvyk98GT^P@0)Z!^9y^!^H8z!qzl%K$=;VhKVC5%Ph|_E3(YWG)x@((#%0= zm^dEJGP7yskzH2>bPJbIxlvgE2dY&oBOp!BqoU|@2cD_>2eRZ{?FBVVdME+vE*#5r zWx2_U@`d_qVdvGS=#G56{vX)i<&K67rrWwRYV?@&C`rBB-r^>;b@m!*SGAx}prUd| zj{^UHzu?ka`M(JEqF_2JVsEE(1HcR6&4*A7RGjy^X7(VtQtsY4U26yWM!xf52i@%4$&b#a(1b>%057sOX%Sml@p z!Z^}(T4D$g_d^q&M&Uvq7)#Zc&Sc=K;kYcXI~&fp`u=d2A;7mPpN1U+A#<^3cogtU zz;Zwgox5=t=Uf={F5Ku7F5DK*+!mmE*L{*QZYwt-6#V0s46u_7((5BRF8BOM+!mnw zrqxKoAY~`Pjr|Gx5WD$4S1*?H+yWrra==LT0f!-Xn)>{RMg*ndl=ObVNMnRQT;oXU zD5X7;(sEmfduZShwE;1#@g*^pY6aA=^L<~xqY2XO*j~CvdrEY?n;wMuE|I3<5VxTT zdju2Lfg#mOY4{yBag8HsC>jR0|D`NziS$J)13kuwwLTIk4RI3&*zw#)49BqAIrdnt zG)*?b5*n-0TI7t@S!X86yac!mxCZD3^Z-C7wlt`As4(un z<68M96ok?*m22fepJHbETH{m9O<(^|CMx%#uWvyUpdYuErjd{FW*eXsPz!h$AV_3H zkc0z96G2Eu3;z!9$$-a+Fm@~v2H_rOCC7l<4tm-E0>SNXVL+JCEYbZF3*bb?8fvtS zPUH+mi){>N8;i}F(H3T95<}~{1Os={W3f#(+6l7@uNTW?pu?(Ei z#&Na?9qd}$bcu`s(M{!S(>UE|b}MHK=5%44EtJ!Zu;ZaX z^v0WYoGnIo44+eU@8ds2{14Z)2+?uVf(oKY;52sU%>4R zMA7cq+(VyldE9Tr##TcmGPEdb9c=@^I1z}$)#e>S926gm+BdikM&sPzjJ2w?i*^CQ zvWC$A2x5nlAn2@nSW7>%siRf8paM$n@RkA{q?+?lHH@M`z^T;4?qi)K)9T}cVXD+} zrBE~mrT2Jel^5S4EnZSz5SO!PlD5+zIoAf>ay6g=t){; zOH{3+fx9@QgQ4}($I=wP_Hi>kQ*5qoq%tiT! z{>Ze#R$NytHiyNVdrzs(tLCN=tYg-=`ziMA%N1G>!JVOdOEvewciAHensiI5`Y^87ha;A(hIu;gwGShuX|n#F3OFz z+9pv))XEVsra?A(=>JbWT77IL)KpET;iE?=enKu33OHuJP#`G2rT<{YUU)%JT%-Sp z4p=KkT$F(eBeQ=Kf(Y0BPJ1u9Y4@VDJu4K9BNv|i?}4V?7ig`3OF0+T{P#f7{y;XH z;HO=IeQdHO-mc9*RL>~h#1W2F!eV;e+{@nf{7N{pOek1Hnx7Ezlwu}WGRj2$>S_8Q zc3r-VzHjtgcvNW04bEIai(=J?3y%nSGGBi0Vq~tl>#fZ;U&Q^f07^g_yp5DQ@D^wJFJ0iK|w|ZH#Lb!n>|{e z<(&~|y`5rJZ3?1gdxBC)W5WA4o~Dh#D02jVHt2%k9*OHj@5vrYB1gSZnWKiIQPZHq z1i@el{rsJ0T`t921)0O(EHS5TBHpu}b5XDH z9N_?46Gd~y(|I+PzHIWKcOhqd?-29>Ug#NGw{G-*#r@|Hdh0zzvEZ+G5H%vDomGt2 zUtG-GKyu;2e~p$uJLC_bye#?rca8vJ&of7j{aMNWsL8*CA@RA$NDT2Al<1R@aGHfY z#c+c7EcDo4qy<|f1FGFV+K(#zPpHy@G54X${}Za5DqLIt*3i9}1AOQ-%R|8zvBCxs zFM90%D-sc)IggR2{q(y0Lx-4xa?B@97>CKXB2T$&)zFPqLw2YRWud0}qL*~k2+MxC zVY?0FJ5^9`7YhygK3#O;v>|)2-CZKHTSMGX_PR|Q<1~Lm)z?ASqTQ5FS@cewsak8o zoahVWGpNvcFNS+Y3?_DGf;Z>}fX@aWlTi)}IF(|GUP3j%F6|-5{06$N|0%HdA~lbbU)I-zOQoy*k4FLIjQ-yl>gB z2Rz9@@YS8GUG}Sb>35x#b_8bGqOvQV+jW&S#M?5vK~~Ody>Gks>8cN$eUoMDJpy9r z3?lX;5w#7vrlTR5N3}Sfz-$d||L&t&uY^4;AxA>IlAvdGR2$@*+IqcB9hw9lSMWaUeR=Y1+_eRB?RBokJ+`7pUTk2vNM$9&>iNEGot|19{slF5XTt`!mPY0~{c6qA1t zp96``BYh#Q;rE;`BFt|TH-)?DvcICyJc={LyG|W&_?6HZ?huB|_ zFxBSk?F9R;xQ$;?$7n1y^64;Sl*A^|1ZUEbC{f<+3URyzlUDh5Xp)>MddS&$bABIu zGh{D6Ebzv`G!Ma7xsn~fXbHtgTS>oN29ov%VbKzrCS3T=FVL9z95JBs*I$61Lhy27ix$p)hD zP=7X)$&76h&zxf2`EgIsI^pJg>z&gL<)x`l^qC*;3xlTZg*#X_opUcwif|jZx(jxK zsw&kQqZWG$Lx?s;i~Ij@il(_PS8{LtLmTZYR!*MF268Ql+O$$Ab|No$iWh4am&WQl zxR$=8?_#+*iL&#IF#-pyD>|jb|3Ph%{lI~3ucy3-`%u_(KSo*eV+S9n;{Gt#evC4} z1{TE7(q5@r6CXPx*w#pMA=aH!0WJN$s#GM(I^-)rB)hN?gRw70zss_JacTb-mvi^K ztfcX*`4^YX|Kf7(Ut9);U1+^cXdXT%o<;o7gqVl{&)}G$2$P!MTT#dd!4Y8|cIHg)h5=-YFB)*?bmhbR#9- z0!~%-wV-=wDZHF28iO;~9wEVLrQhgxY*Nv~D}7WIYU6!A&$mSfxUb`{9AB2 zfB`Ej$OA7Lu=2ND-H5Mg9uRHBkBX=BG79Ej@*pf3&VvMbh!gGShz07Da&HmXBlhRP zC)1nRCdnk&*>A>^_R9gm@dj;u*ytA1tQOJ2s#5XaV-BX@FmxlD-n!>6@|0lv$^k1% zzHqdg4Tt;t8oPhDj5_&&|HSSI=;MSy0VtkOfrp`(C`RMGoqkI(HYcjV{~6hYu{tIh zZjrwBRrxh;^GMwPD@SR&op$1iu=6_117yYK&?V)YWsvZcc0$`Mv&3Xp7}xm=jZF!nXucmI1+;> z+F9{h$=&bLM9FZ7GceWgSDHT&ao^8n@GDsow{nQGZGA}GTEcm&I^2>&UZ%rMZnO%g zKNIN~$sHu8Pg6RK-m3PA?$}-aSVwZ+7jHBQhrE)K+q!qf*hivq!YrIZ+;td6*8NEU zdS-kGRhaCXRYvM+g}QnXnJL*)`*yIWRsh?_?cN1z0QiI?I>%%lq%wJRf z?}$?){j09KvN&;PZn0aO(iS&P7M9a!vT58>upsDn()}(tHNa?jzYC5Umb1%-T@2C* zTR3hx)|+mu>?j+X=51Dkc=pLM6Xx@gpVRs7om=b!PUZO#4nq#Rxp9P#$a6~_?(9OB z95&)|=7?v4`40E;LRTA&qaje7^J={z+Uh2Ia)dUi!_w1*6FZUG}((nwAK#G!xGxTlN~VjV2AY)Cve1S zKuaT31_cpgW77V#maEk|bf`~ozUcc_UCS+ zJcXbX~KZyQvF$h*|+WYWT z-Tof$cw8((I^V~wxCMSVtteUz+SB=s{@MjU{ukiKYx~Uib}5u0Ps1#|#V!M%-*>K% zg$e=kPmB>cl%wC`##KUb1rAP4bx2GMqNioOgw~gbE`2zgqVkSlPPt`|iLgWx+}gH8 zX0h$Gy0fBFSw*LemXzB&nho8dIAD%-?6k+`JIyoKKcrwIy(fp#+(UM(D<0oDlV{&( zeSkPeVl;6WwpP@2*wnYn&=kDx(fn(l;ia#PZFiy-L$J%=y3OTv?yS-5gup6R<36!7 z^YG5Ip0KGTeS3hDFA0x|`mNKPYhcp92nb9@WuDs>H|bVQ0J7&bMr{4e(}^qpvc%N0 z9xTl@7dy52H^$x~l@W;}wbjjuuUGF!bH}FWrcFinMN(2X^48h)^)*6PtdE?W|zV8JAv#JwSr(}t*Bk_tlU)tq$PpXj=Or6DSUF7b; zc`G`~Qd%hGb!>aNiq&9mL<>a)ajMbt_OR=oX|m6YD47l+??55LwpZR~ZcBwFyj7Cx z%z0^}EN7%_8g8S2&)M}4!S-1%`)h-OrAIxFpw`)U>5&a>3f}W;bF5EX4r+ud1RA&R4y)%o0XHSxUc2cXEjFdTyXrQ=Ki_*^;#GcUe?LN7+f&&i z?vZ-1AI^vT$Ra93!V`W&?~OZxQF#Nab)z{LigQ97%3Bw{7c^9j-I-fLkp#+wSPc#k zi-~znFFW2TLmnN5ogl~x6%UrkiW&w@ik3?(=5Z2?^0ARrTQAToU;<`EKwb{nfC!TL zm%Urmj*lfS<;|8|>qte!igBYryYD=yK8BROLIvTk}mFM*0nJtyI)h@ceh4sv)n*Ssr-XE#RZVtXwd@d^7)*}D0C zY`NlMdNIygK7#4-2ON*Bds0m&SlJ}x#9mYjhx(oCaOC?diIZ{CvvRNOX3NfX;D(Uu z2nc&8GKukMOi)fJA5KYZOL1dpSpkh@QRn(j!T6Dcn{!hfEBTSVT6d4X9gM|d2+@+a zsmw>^6|p{LMI;mRgB_q<1pT0w%S5@dvJ+SNcY4K9e0aS_5BYsdJRI_i1udHXP~r@4 z_!}}|-Bc#5?7G1B?(>KCt@?)B{ntWSG(FmZd9!F=7i zlRw`xP3_G{5x--NP4@VReWOBjJ=xlb`qhU$VfvWWE>&VvEhcm92MS5y41EbOdomXq`>5i4OjBwtXul1`E#E#HKx4TLWds$6)n%xXitQm%!eEtrpk`@mGsCupFuB18D_gEQouthyAHjdTJzA zRTtgz4!5{vrH9@)3&(OLm;Gsnn+gxpMGJLCitO`Ip(?Irh<%S;sqNc}+4dJ0YN-e= z-Iw|g+J7Q0zQ6#)O<*c@ES|oP5FHyw+;mGI(tHQup$4GYJT}SFf56*!oZklPl zUiWhT^xr@Q(j37)$#FTSf_#Me&+dN$iv4HzB|eF0|JnWZk&pyfc;DGQZPqW8Z8#{% zu04tpBE+=Lgl2B~NE_4o}E2M#0x;QxflhNWSd!@zwXK*VW?pL(i zw&r=VEuKBv{H+d~jQdCe(GIUA*LyP29piR$wdSt{>@ZE1dhKuGOcuLoo-Q-F$#SgE zk}5M_fCz;QUUo-^&wX892fXr*_c_{tpvX%NaHPt&o0IM&4Y1R^o)zr84TX~&yGY?d zdHv&Lu7rGLU^*Tr)i1b4lj5z=RH#`zB^jFcW?n%FHGe3a`A~R7{bV2?Bz_+k&a?@9 zJbmY5#s3e4<_};gv;NEnFb$+};fF5MxEh2rhXq#7M~dOJ z^xvDxaH-Oy+tVaT{q=6f;d#n}8+zL&Wx2?TYPl;-q%Vr8Pb0-Sd7)%=cHUe{9|uM% zwgvY@pwR~TYu|A#MpyV+EXN$-zlMT1qr8-BvU?-Ej}Z(@yA$P9CfyLh`PYV~Oxgov zWKpNc6NnoSV1fYsz$t~;kWavv%O?E8w59027dD@TUpd3Z_&8sF%a1;$xg(444^liD zoUuFgxtn^#m~@p_uPw7s79F}4>NhABr~OK=7qX< z1T+V2{<7)XIa}F-D4b`{q^NF2on0EPAh*7aySKTOB2Fec}uw4=|v%Q&F=HPOxbE$7ndjWM+IRzEC9d%!r2jDyJp$nPOf$LxR04@Yh_2?b-yn=dDPuP z|2#6>FmdN&=T=*tU3cMTZFQ-4e3koav->Y0?#{W5NueKUw&ct9~p&5T=J0V zSi^3{_%hYEhNaV;oSm~)u6=AP`=vH}YuvG3K6VGjg>tE%qc0Ud_&2trR)?M98g4ow zz)lcluB5DAPAZHUiCI=a0>DajID>WD#hLZA7vZB^W}Axk9Z zsU{y*e@K%0La%xLN~F0`ZkkKeOp%wgU>;_vxEl>D&hWVPv`%~*mIy0*<|defpnSiQ znR`Z<+a@51TLX6^Pikapn7LmIspw}*0oi~&01uej1fHH<5R$O=hi$Z>)AU=A3=y)A zSi>nF@GTYBQsw?@AMXU?J`ZGI#P^ZQ3zr+7E(Z5~%{sR7P(Z7v;qvymXG&ww7>a<4~LgzJ$xfeZ^fXT=?u0d-c0WsdC}wU@cg;Ob8ax8NHYGL@#n|4Pk5?E zmUBQ@czl0?+&i}ygIM0&-aK52_q^8OX>9R)f^*qC8;-tIS1*13@yTNoUa5W~XP>Za zvWS06#J}a0e+QM}tX?PNeIn@y=yiD7i?`K5GW+}~<#ieR8kmIac!zC^{v$|bil{jc z2ps{pT<$i`^U0oFF_Cp+fP7swC_TkcWT*|@480yHA(nvNxa|WD-4?NrUnm|p^Pm7E zWr~EZ=pJGu8xP@k8 zkjHHoOs3^_7#{@#GExh-XgT~+3xtL5iwLpF*>T&1&SY>(DT3KSC--Q|H%gT0>_X~R zfpRo|H+%*rGj9=eF|<1*%BaV4cRef$x=lff8)=HcKS)uh5aJ%~hqsl%jTk&di_%sm zE_dLE->ZZle&^o(_U~I#8*g}V5mRn~#u!naasgf0-K zEZp-|hMXivr0ElJuW(6A6cvQ!^40r?=8^9K0jU(AXbT!PZxVo5TK~ zC)Vh|o>2FT<9p)$n;h&Wxu#+c(oxt-I2h*c?!);07jy$M=lF;5{w>H>aSIZGQH|1G ziav(@H0(TS-TBbd!YL&#sM<7R6U}f+DP|g%nc>XRQvo0i;$UZ+p_rp<%IJCtX^}`e zB+NyL6oOT+;d%>ubwly=Vm-$YduBeDOkNHAlOga`_D{KrSA$l(CwX-Q=%0LmapBY# zbFC9Oh^?4|FHJHC*-zIT63LPaX_EiL-rL7Db!Gd*=bVI-kQ|i&A{yG}gco^p3fQTj zcFJ36)OH53BHH#2p-R!#UduS+%jlgXws3;9g=i(Ac6t&;0+kkMe-*XN4cJnoFr(O7 z=*ZOC!J<`1t?j5)^L+P7Q0#Q>ow@fu&wc**_0z~X`|Ovs_g;IgwbxpEZDIL^JfkL$ zt}LRfAVR=13>q1X7n&$C1}{FGbdQG%DB}52W6;Kc#T_ZekmSb^QdX0S!?Tf4bJcae zDL$q)Nh&S`f=yxQMiPeiRh5+&E^MekHBg~6WFXMlY2ovv=E8<#=s(;YR{BJT4ncD- zcIf-y{v2sDhSupr`!yO1uQQ#EM6k7njubk|G!*H2(<2ToNXa4E_w}iW726LxIv=_{ zypyGzzxU3eO;;kRp~!Yc^hriItX;FD03Fa4$LwLcJ|Z0EebU#@sa(%$bQU9=XAGJ@ z?E;Q_76d@)k{5k{r!lzHIAQtcKo7_;V(ulF{Bw1M5aT!*&X<<*cNXz?#T`!T=rn5d za)iz#hYd}7kMMchVT^%=zau{0%3F9u_a*&|h!-w2B$FOq?1+*&8Kdje4TpxhjGRjo zjN_LWEZ2O^t|`ja4!x^GH{9=aN&R*`NVzmYopg*VpGtC@Fp8aXsjw0Hgo5f1gX=4e z)-P5EqjktwaWr)4#@{AOYs&sbfEJn_8n85!L)2iyGPDIW>dj7%}-e>>p<^Sx3P$hKO4;O1= zQv>={2j|Sk2Fzicqsch#U6&D;yqupyr$km@>QGV3Q7`h2iQd7Oo6oz50RsPT2QR_(o^^YOMtGgs)@gY}-!F(rqOT=+EoSfT2gwaH$ zWqxqWX|KgeS>}f%K4cM1*vHHdHHVO#X6t!woNIuwsm6&tx)WmXuU^bN->%u`DxR!J z=`ifufzUJ%us^-*`q}t>&f>|eYf3PoRjw%p)AcIX&x7{06i*Ity+VOKxn8-~)L!L! z6dtzf2F}7Q7j0W?4{;_(mCK>mi@~;VeS#o5JhlQ}t<;cYkBm=W_gOIo8GvhUB=@%70*WHg40fIL zVn}$TJ29%IiSD6aa(+hJ;ap`xYsW|e`PypBp_)W)O zye;RwyC}o#T+wpOyWt={xYdQ8P$-y^szUwRByw|!UAb0=`D%@&*;qmHsy{HJ+3RlfUExP2rQWd0qgN&3f*1@&2YU`*ojc?EXkbnGIhhy|p zVp3*QUnBvo-GRlyp`&6m9AM!~y35Nd$lskXSd$@;Y*=;%*o7Ad!=EFq;a&2a!F7Kx zd(GQMlL)*>s`lHmZ{@n%@U6mcDSnu4Zwdb9 z!lv#ud%gBv$hSWkT@RS+u#4l&zx$W-(>2ZuUT|{)kjk%B6g(-rsiZVmVy{+BC-9`H zoTt1gZ%}w_QzgwBbtY25`yn}vXhYnAEnj$(W|HptJbcv7Pj7%6=8xrI;^Rx{=(`KK2%Xa3-?+?3G`ow>Ci#8LX5ZU}-$mnmD9LvV&j`=gnl;&zYChU{~2PRqQ5 zBx2GHSl^G}my6#+_$|e6gvbN1XbE$x>i#`E2n*OCIbcPO4Xr* zmg8>Ak~QLP&-zCXU1>eO79ZsB=)BKIkc`6qek-3S6MKwLSF41z|xekh=@OSkF` zjcLEmd@uQqZ`w~+Jblop;p-I*P&B;+DER$uNDd zQ&KfgB2Lk3oT~~iamwWxAb;FcQ;F`PXo7y-G;o;HUL86#u=>z04;r{)9(>knjNaiD zj3V?YYQzF@c|SjTPBg7{k8jK9jGt{ao|=yc2^`fI04Gmo*@)Lglhe7z#-WkqI_PB~t*Ho=q4LMCPUN+~KUw%3msu5lBf{1SbKyGoCE_MMva@t&d z7PTvK*(+q~5Jn8PmbzMRQ5FSJ81@0k*k65Dy@=R0aQz84VN@B9cN~OV{pTU1NTf!2 z{x7<8&uY_Kk=S7^6G?5T9zo05K3dyScMb}JGS13}Xzp~~U%lJ!>8I&Vnu5OpL>I`C2;49{i#hg19u2J&wJdV| zuZn2ev?h|;78xH0fwMU4cU_Z!;?>We!#K4CLLm1YmUS4#I&w_b#=3BBuI33=Dv?QeIG3H6h8Tlbzs8$oiU8_?fSHzQy4IQ0Ms!6@re4IG=;Ta#$0J%Up z)yyq?XFAP&^Jog$#AWNUwj@Lu%yFM{dEaK!oF2Z(#J3>isW^^Uuht&6R_l#nkdX5D z5H4G&PH|CloYQ2-O6-TZ_w)`b#X&=g0IkO)fb`FC3e`mr7Q1O{OSS%-w$l*k>ELl5 zu&t$Y5wyrekt3&CwCA)3G83p-BG$e z`YZNviV_eg%X-0P3^G)kubj=_I^M2yX8;*2#z~oc81TWuPpYqb+^%f9Af-1q5+p(5 ztE$#mlr{-frZ)o+-_(*Y3tC>AaT40I@snyAS;72k1JK#tf#@@}0=B8hZM7t9MUnG@ zh6+qKnOd;_!BCn1c-B4NuJkqQZ!??TB$aCMrah7z&(jjlYsfwG3|7Uu!kL~QX2<85#z|c}sn`xE5%ddb zKKrb_Wi6vM*cMW#6zcHOsVn*d0R#*@P!<5NwM^$k8maAakOT!)91_blns*)|;PJtI zY`bjPDQ|3y0SiA!{hVpl>UW|TR!4CPcnKMxz$2%7L9MW&P^`vZ?86RDJ3Nf5z!Pu( z{?cf*G`@dnKAQJ_5U;KDm|$WJGj6{<|~oWO5dmR5d%eULRs-x!|b{lMs3!yef8!PFDyT8=HKy z)@$;Nb7#onp?r1E4y}(05yH682i=o!k%c?G5lWK@>3s9KQrNkgfC0?^;E2OxiRz^4a&%kX_-cQ*hSD>IH>6 z%tsRmFatHX6bQ*`kndpq915pracF4I?S|TC`CFUq2TLpi*8&NULqIn>xKST#SAp5@@|hjY*+X_>j(tlr}Q?r8B6|AZup`^2G!7 zQf0tlur+9u30ouUINvo~CSb2@0MGpye%`-$y*Oh+JqFpe5Wh6=VF5qz;TB17g_9kq z82F{DBV2FPq!@$qjN{GMi0>Qrr`%(%j=Ri~CtMM);h#CPdsiwWIq1PrYc@ntzh64> z--O#zh8Xz5CeI@zE6-ZSjOwzq%2O#Nm zOPFNs33$bidH=n4^qqhB4EMD@c$aQhcKwb3SI=?iX(69Pd_8psVw`v|KnPR`kC<<7 zK&HbH04Fv4%?-E@%~$H@2WA?_Svu~T8Jh_$t!Rpmio1}oiObALI}_Jo%3U2{v2=t( ztXfe5qd@nCB4bdA97lLce%0!fBkBh?-uZ?Xo!gCw%XPo-mdH76QOt@ab*J+JzK(gl zLNZTi<%5Y6gU$nee%*~5CGyrC-lCY!WJWE8&wQR}ftJd&g@5p_HpW>EC6`fTT z#XQ9&oG^w;nX7GHW0m*1iVPErf9!#Ug~?S_$tU>G!|K0utSfVF_B#0GI2O~c7K~#>w-<@xOYWE5)hjNj|=>E#@Y2Q`3LBg*%lg zQgffc6sG5{jtMK^Sc@3Cg2a?6Y+k*lHol|%D>v5}Kg&ux_*NrbxKoOgLj#SdPa~=B zNIT!A`JhSmxcUVgO0L=V9)xJ!LQx3NEfr0<$<#uP$@HA*d7YKjX4h=pMnIw-Av!;v z@K5j>#Jq~@05{UpTQg~Ur>Md$rI2VL=w=$jOO3(iXtK2U?O$-{cuXGUtcr=Z9L1L1l;&&xb=C z?4r>hJs%3Lm+7RqiO7*TKNMLld>AL^pZUrPv0y@`aeQ%PjO#u6e50mr>vbE?g{`Ao zp8SfNP@q05!_h)A!#)}ifSXLZ4Dhhr0#>(VtxG{cQ(GJo0Z4;>S>>8EzBG_I)TlA7 z(X49J!TA>41ABCSvI~^FQL~mQo%)Q#tih>^Lnd~HO+qATegLD?_3JE~S)I$y4-sP5SF_4U9L z9YOsaG1zBWj8km5SN0j`*3tDTxbhV++dnU@4#m2L>af`=0$%O91|?Y^pLb25j%~6Z z#%FvhyzSdz#8haN(&2C2%12e0o7XfqC>kb}22!O4TDYKHyQ*rnRSw3=`HBsxv!25yZ9(X8U6!Y2{~0tq>r+as4axB z614Qu>EesLgepE1l2O>RQK8Ufh2v*{&IulT2-F_{r9|kZ>MUP8-Ywt#D~sSdEA>2V)zcs(|G8vJ%x=LXy7VZ#eYvWJ zg*3N$5W|xrF+80|y=LcPPz0hl^68SoPRa#l0A4(y?Q&STV1RkwTliCAm!|^K3yHtP z>g~0KOM?`}p)Jpn%A5DKHZS8*d-FS<97q&c5`-%1^&uD~pazxkJ1;MgMhfJ*XwpcW z+_=yC7*XXISU&1fy5RvTG@W|DiG;OmQg(l`?qS{3prm_(bUxJ`V%Thk?ZFEkJ98iC zfRRexlL5xattXYrhJSV)wgSA@J$OC&wbQeoGC;$W85Jb?rgIYriy$W+OxzTVef0Jl z8`+0hK?Z}kC6@3C>;V7eyOy6zdQheL=ihO6pjL!rd6c9GL|@jXHd#IU8pR%q$!=+r z)ovFXTTMdi2WjL0N<~FQlO-Vo$82%%4!)c}`|bt8N8H5MKy8c55F2oZ5EzABwz*N< z$%AV&(0Fyz2+ul`?tNr)N#Rm{qbfpJ)25OP#xYx&vQNAhMe6*Ux|^`qRozIH&3-Bc zJ9QPXBYEj*oC7S%`lK{#V0#|_iwx&JqU-Bs&MJo3++Gj-zwegxAVk-?{pAdN~5550i*M9CG=FGffSkBwq6aGNA_igCA6`bB{h>ivGoZ0`FhiO0U5GCSf{ zO2i$gW;P5kqrmtQ#U=DC^FDbDbvCF@TM47_*yWlu_nQvAKcCXMkY z`D^s0l#on**vF*!A@!v!CwZRs0X8OO!`KwKsx0Ns{7uTWu_;d>4+*G90QJ=)rQ8P3 zN|NU`c%+ou;31K0BW{C7O1TZ5XGosg;E_^p!jlR#g(Yq&xszseFeOP6{;7BMi$K%S z)PB*8d^r3-4tGcrrGzD&qIxt1?vWY+V>A5Wv8##8TMeBq!vUM&kR%@v zJQq)K_op&L{xFx2HrFkpUsE z7Cn_*af+Nq$vTYMe7UfB&FaVEYnAOfrWF|O-{4<+T@6v;F;a;55ps_qf|suj_~&Lh zC7CJ%I)&BAYfilz>~j!W+RpWh*G64?E#ih0{Zjs!Q~ly^5VS|njKi51y>4I&PO&PW zN|N=9YcR_Qat6uX7k>*WWUeB7DP(8z&~d8p!fXxoHGb#uBXsi7Q(S^U)}+RE1`L`= zK{EXH?W||w6>%&>H_E!QDYwF7x^yvqMW2#LZG#$V&prXka1J2JI8p+k0IqBC=C^y% zP3NDzE$>_;G)Fa{X}%WJ@i{KV@%(}8Fb)#>*R2xh=68cShKN|bpCs>9wfyQl`7&V- z_gDOF^JR0Eo)>oqZTlm&VmdZSI1_p``iY)Xmcu^!BXiHuXljoL&*;PJHV2 z;1$wy?}C^Z1svJ5#f0yKz4JogOnuKn$k(QYk=h}6Bak3Q2fe9^goW6_<>vA2atEL5 zx#Uw&Do8kDo^0*pI;6#8!tJ~^TfdmfTTIV$#M~3i1z@bE1r&)Jh63*6%HF}AFQeYe zuO;2~`PJ;Y=~shEE-nXeg5Bi$2w(d(oO)rJ%fF=LuZ|QTI>gh*5lMncROdyPa#!&T z%+cmaUT@8>6NQc62@SeQ@=m@-R7yT*E>SK~kS4j%Ej{$Xsm-St!7O~?t;4-JOp_d8 z#~zm9`)J@X3q(uHuPk!QBkw0<0~r37U&C)m|AawWuStcMl0O=O9ttjb*q4ra<@q9^ zk=P2`-v<-b=0&Z@jDo8ik(qr?d4LbEkT~Unj5L}`K$1xy!Q~ljAMd)I&Dw0-7f}4j zr@-~UaessGnMmwQ*xQ@06Lo&NjZaD{eD)8QU^IfLUTA|oSck)y7-iQ~DoM~)n=dC% z9)vvr22p%My5&jA{Oijbh;Nh~8pTl9uhO%~hdD|vo(oOSOfstome$R@4y_kX1G^Sd z$(2Wup+JFQKr{lAo=*C45VY;k6+EUzUehuBJp-fGYrgtNo_j%G^*Yvc* z@dh{lwQXwN*WdbKQ1k!cTeE4GoccCJEtY$rt7=~8DGQ}$zS3eGS+Sy94L}j4bnmlO z*zs`CMyzgoc2|5ranW=f_-IqRmAj_8l`|Rw^xb6ZY~h(OB@MjK{YD2f4m|7H**cM$RgOb!@2vgtu8DG1hHm99 zteW-9+(kQbOZoUUu-|NZkj@j=u89uT7QZTsDQ0s1=$)p(JQMG8;?)fJ%@2M9I#B&0 zp;56)c&^@l;->7o(GxcZGls8R<>e#Y8Cb*#Sd5r!3>W&3UKCH7j);N^zeB6ymP4yO z0~aDGuzn!?-CW#~gDLUgD4rSyZi9*}v4N-s=cjghb4W)B6KawnQYTWAS@;D5Iy>0G z@!6HsJCV4YBLgP8_JBUJyRQ4lyX@PmDKy1>gRO==V@90}BTsFZxM&Ara#ATY4I3Te zF4r+EO+D>PrI9@;DO=p6!DpDDOw)BKk5I{HzD7F^S@G0B@G~soK*;jR-b?&aLi-Fd ztRejMp95Ixqk9?u{^k7p&+DTMx%!~|JVo-mUqLfCE_B^&ugk%IQ= z?1b~5>9E1T^hF=Nj^h={KOG^%#Bg)bHlF4%(ULk!U^0I7WfBg4|HbQiWToFmj$F&W zy5@T18L0?j7GUFpS-@MGWcHP8@g-mK_2(>+hN(hmbpKVCm>Ui6s86mU1&2E%s*bqc z?;O1UN#Fj6z|~AtB`}@>Hr%toO@WgtyR7#*cag=OjGR^a6?aLOvuT=qrA)g}{{Zyv z)V+~85s)@O*NbtSIwmp{M~d#_B7j8FLt^`QqB}StV%QXj&kZr@DWD|+VSaYaPUHTG z%2UqIynWb=Bit~(>`6+{l-J8q^+u(s+!HaUykUY3E_{~A8t9Jc?SlXWLJrQKU5v~~ z{_^P0FLzz>b_P(+KX}{OWlvDC@{0jb&;t}MCUT+=lrkEeo&W_{x^BXfbU-HoqrDes ziZg66F-%t%48zL~U&ox45Zw{@@}w{M84_P6_ZO(hZEEdT=ZAdxnB4eR+<*7=3JVk> z94?}X9R8PUxB+84MxlK?_a+$uUq^2xenud;nfNp0JUfBgh=5L`W0IdqQ*TG=PUx^w zJH=R+Xc}(vis`F${Z$rcfoaHe=!l8q+U5A5)#Hkxm=vaL$XmDXDq+>*Z?bA?1DVeb z!i&<(anItU#VXw;OshIOZoZyg3aXRVlAg4BarJU%oG~qf3z*Y^Aakka6_KeO7KWO! zO9fY1lisr=4ru@0`q@)5Tlz24xzWkuod63e}D|=n~&LLaUD30 zl7Eqdo}0TI=d`v&baFysS)AEg%`xU8Z>NaJZ6=$7dUowesv)vf@wLr3E;{WR@wx0B zajrhu9_Ute#-}@{p9Qn#0v#HgV!-W$Bd~)$8*ouPKE!s4G*sa@!Hsnjzr!|{s@-OL zZ~(EL78a*$cgd)NYAQZ*kv_n}E2p}D!vFkaK}#$H2i90u=bHnGn}N$%5PCWY%ppEr zpH1myl&JUUv+zJm5AY*pM3={v1yX~2$a2Sd<4;dI$Db`>xu=R^(j!tpxf7LEXIx97 zz;$xPR&!i+j&zSXjuue)ptXyA^q;@~Wb$`j3vN?L4a)yTKwbIOR_1F^i7CTz*UV`r zomb8XxPJ=_E*&)EF;(<#?_)|!=76Q2u_)yx(>*ca;|`zk66&A%O_L1gZR@O8Q5zz! z==(|Nr8EooR!Sz$=`KNgg}yd3^e1`MZ&IQRaB|Q8MJRvQ$#z)*_)A{2#B+$q-411G zQN)>*D#C;UV3TZ@l}=sCO^fJ#TuLHl%F(1_B+uA{GouOIG!$x=MO^bG46nPomWjAV z)SOzF6iAUiRO=JV$y<|b?I%daxTi9uqGwg3m=gJm^v0?L5=J-`vOV@4(h(8KY@;oW zT#*t<1Z^CCA0ud|e8iv8IM*Fza&dEIqH=Db`|m1^m&>En7K3J4i7#;XEyE=^cd#vQ&V z0EZ5L(4_$1dy^Mk!ed&1Hv6#!Gq!HQGNYj5e|FNRx=tNFSRni#fsl7S;GyrJ;U_6Z zZSBdcXw)*b5!Z;vb-A`!qyifaN zCCc8T(k0VK;FvBL4|@^yk=#kseIS2|clM`Ak9q3tX}H*%cvi{GmAlc{?4dJr*Ro62 zSQOMJcC9(>(MikIuAKvBJ6SO@-V(yrjObmFVdgtX=PX;;W2w>? zB}to1($I;!a8(+vR79oGkWmjH6n8gRxe}KwDya>Z#n>IFW=%<8TNZBC97oEu9oeq7 z9LO%$Lb5e0rCOBMBdw9amHUw(_a#`K2qrZqllbZj*jufBM*=gD=A{Q@8NQp$pwtiZ z1@f_-!KJId&Y^DrdPWJ;HIw2d=Q=c1cAaejYY#a&S!s(5IysqPGb#I9VQJ8hnkwaM znRG$Zz*HJKBPW&xQEi)`)4GFrMg&rQs$x~L!0hg}b$=mVNf+xfDy^38(=e^qZmCRL z$FNBhy7^EbHku|O5)k9+yaWg>A?(zW@j{5H*CQOMG*}{o$kSV>88Rwh1$4O4&dx;q zv1V3A%LjuskIusNz=M|&<_BMfH6AXoS7~41h|Ga7y`v4>v9OCQPZT>jg(*Ai zZ7deocur0_;XhN0d+D-pf)+QoX((KX1hiP|L*ik$FKuDN3^#1Rj#KKI5eW^pk@j&5 z_*=SYTpSpQ%W+3BNBaB{`7Eu`4C)xu@zP*R)u6L#kj$-G1G&)roNvzRd^f{M^T;XI zG3$cY=}U_yughGQx2`B9kIl?eaFy6KyE-2;zg3Ak&89);ra`OC{F_0rBCE^%Z-cKY ztgYsagK-vXCl@Z-H#s+uPvY85s~oyshjX3uX5CV`*40^HUgHN0IMT6Vux0h2a|MBF znAS+Tj<#U)(}U$&rGNeQdE4~nCy}7=E#F>mTZ(zvt@jV5LyrXCZmk@t|34+okdgg? zZ-z17%nJvr7Y;gqHt0|S>*nDgvvb}c7&*5Ydwe`{i4h6Q&XIMUK{+29G^hH(C+pW~ z7&QNQ%sX!RGiMAwjy)2UA4^4F^%_$KV`GG{984%xYGbd}n`aMV46G>_VP8oE>)|Rf z>Uxa3G~}T180p1iq}TPYJ5gE`Tv`-DFan!^618x&7bt>zW4;vfwolcQcn{!B28<{wErGh=?>jy5twCGj1OuFuCB$y`; zVirRN1!exg4Rjh}J^B`X!P~(MwuB6FMD`y3GRTih9oGk1CLnS6WY91P(?ax?U^^hc zeqbfy0`%jmqo--qTZfyUl`eglC66@^58wmR zmLZKn2p)@GqJ_brg(QplQ}W$>d;t0ZVBcp0d3g%J{ilJ_JQnCTG(ga={K*O2G)Qvo zOyMp<|A!IH;(OH`B-IeOaW3Hic_!2u1L3FD677}`1rR_4iJACmT$?ZNki1Vu057}w z%}Cj!;^=9z-ZYM3cTu(wcE<7$TckW}kh0m>wem@)bU8Q#LcxgC&XA&Osj#^>*jK1j zMaiViYgl66o&W9tf={zyyKdNBZ5B73d-RDu8OR?91^XiEZ=8-(CXOhv*pu(Lo-^!K zEDuF~6~}S`>=FuUvQ*U6h0Rs~Cky-f#zhJU{RvL!f5#E)@<{;)5>8H7Z6gsOq_Nkk0%QUYfZ?<68 z_2zGI(889-3N&{Rmsl>smh)7m>8PjT+$m;x-dWk8ex!^Zffz z*sUMoyEVrfJq>y{)6>{MSUBLRpCxb0+w)dJ(lbv$slu9be0kfL!bZ=E{SErP%$ckH zQIBr)?w-lcf_~+)$(&(2@EJXPRoGtg{tG?>8 zdZ`jc!6Z(#c}>^@-eWWS0=UK z%4*#DKE%|Ox?YQ}d!~E1DZO8iejU`1+P0&%WAXN=o?;I{63;?U{+w^v$FTRom_MWv z9Xbs828iNB-kM`v{Ooe`y6QH)WZF;S1jt)5?xjRCOxJ8y9oM1jEeU_H3b#$HI#$ew76|#na$u1hvOdUyoa74plt|YGR z6DfW~Q8P*+K)5!*WNDklg$=&X$ZT@6l!`ha<_pC@7VE4%l=QCJ3m2;DmVpIrn+^gjG|>>UYYHwG?vg?L*}^6UhF$%d2r^}Tgi4dS zp-YX?MaHO&Okcp!w<(Lb*~)CVSVl+_bSakkY#I%ngtAhue$L8Uaci(;zT$*vvWQ~E z;nSZ-C?h7c(@lyKMs2?l=dmLI)w12PX{wVdIjKrf`3P1uEP@hkmDza z(s~YIZEyY&bL87EfDkxjoYgmQ_N-%bY%UW%+FVpk_~@!#_6s$t=K>WaYDc7fK~+_u zu&vuLQVD5Q(%pNg%DUtPWlk~R|cnG<3;tI{th7jg%?OPE8 zVT+V86UQS^@UozYlKe77*`2_yQ3{gk1>Ss6!WKnNBu7HAIl6A=1?HX=M)%JUUtwD{NddER{T$0+J2zBkG2-jP{zA6t}NTZ}rJTYS6|0!8dvHy=X@$pqq;jq+G6(`#+mY@pcSz+| z17A0md4@blJS2F!a0*L?w?k!~PaDhQ?rt3TWG|vMc`ABMjh=(DhLD>(Z^j&d8~%J& z2HpbgWoX-`!otF;;vjBZAEGPzV2DvIB#cSOYCB?;!5?w^JZbx>2G1~uOViGW`!Uh= z=4ZGV9!tM)xhL!>y!Cyos@i*OiV*T->NvZsQ1Gm9IAqokko-}$Qy1>7oYM827rRr5 zVxZsXbeYKujMqwSk&T%7`p15h7qV>PXI8{a0JjA85B}9hp`;jT2Ocvg5W(>Kcym@N zEEz|L%zcu)wlhSIUexDKY3!+$}#?1dw zD$M4qz4O)-fBBhUYhv0`%^#Cg!Pcg%drO3x>x=Y!Jo#jwp3 z>E8R~tyLc+ed_eR%<_Nxo$qy~^liUXtY3O=BhP}XGF4)}5{)*$Nz%s9meCk1FyGK8fv3c!Q`g=Fios^{`$f~h)Bw6V>Mo#d_Z;;i@TSy*(kJvoVUX#EI z8fhz0o3KrI9dqk3D}->ee714 z*O5fC_%YF^8MlADTxtI8$Casq(o(D!!XmDeHv~xhiwY|ZaK-4))SCH^d%N9=4HwN% zk&<1wE6L?lnj^?d*b+ENQkE6zmK7N;Ql6VEc8j?5sx>~H+)3haR_)idu1L>&T34)B zw#H{50;X+6hSu%~D-=YxA}!XOPwMfhPY8_8r5~GfNs^yuISW5FKRo8$y!vCJ!fS_t zHjS{~v&46TbRT=z7i3B1O&qgbzTrgW`v~#D1=|WJi<~mWZj$D9t!*W9W=(|3eD)(0 zYOrlo^i=bu%lE*lj|HJ*`R*?MjD?92dqIm1J>KHZrXT zLZ`_J8yEblB2rM}hKF>A#bo+si6dRGRC&JX5ljeWR&@#6vs%Hbp0_{jWx1d)kD1jL z+@#R%kNyk7@}c3VkDYp{;eFNL5oOk7-` zEnc39ju2cA`*`!qCTnZeYb35D!sFp4<;g5fQMI`Ew><7L0+_;5PMT4F>{Lq495Me| z8su>6+~rF+`-ASt(gT4j73tGW{WxE+yT9*H6El3lt1WI<&Di$U+%+j5T*j98_37N! zjdD2nr&UJO9QT!BsuG{pg9q%|WjuQDbvYI`HcC`Jk< zl>}nvX$|CpyMif`g*@qVf+&-S<7K@wG5it8ygGjg6g^54jtGkPNY8T9b9fGwo*5KV zjIgLitFJUz17RJb5`8+ZCU_{};>u~NW2(cy`^dTfBWOWdoe!9Qe2~jJ4*%rFSO=EfwdmCtsd(%hG#*fHFG)3k+NF_u* zZtV_jo+`ee#<)6^^Mt9Xms1XeCT$Zt-wX9TFFKNK?@g?-7pWZ(Y&aEe+i1{)#f{H= zZ(@3SVMTU&X2$WY8i%H9Iy4qCgc=TO2*99jPpY+7*)rzso;7e?{V>O|T1WkTOnx}z zP+T5uO&*aCOvrts17X%aZ9IPb8@!zBSkZXs&^Mp{Nj(0|=^=z*gCUU`(uhO4!3ns( z*=yI-zOsCBQ3xFt2041P0j< zfhT&87pe>O8XO?ecU}{iuvo7^uQQOqr#>2*ZVFw=Op|RdsV?|PEnr=CEBi$X8eAx= zmUUt7(s9)E&x(R=XPyV4IdTMp-mvX)GNTWEn_k@Q=XW^Huum#wa6o7lR~ zybaNua-+(sx{rZg2y1C0p3+V7^bCG`EI5Gzv4B5AGY}5gW&3fQ+O5cMyxym1#|L;$ zG$yy^7IGz6-(PfgvgDIHlNg|-CrRriiH$DNTO#UaBWZM0R5^8A(Yg| ziVo{Oa^5K+%%9BF1jfC%MbLIo{$T7k@c!Gnpc_)5XgLt-pbw>In|j5tx7iKfHMb@0 zd)M{e@#fZy69>X)T@%GZ4%dW#c|Y9?zgEou<|!k}>w(1M{G9X9Io+uV1Il?0N|k=- zQ1v{)a$usVvPg~avV{oTkFDfEl42?Z0kdRaUE1ji>tOhqCY-RKHe40mNsYTjS4Jxu zI;L&hb02AA3fibpbOV_W2F`O7=Ia(`Vt5n?Yv!W~?yMJ>*i2Yo&sc2Tp}_Xz6`G0$ z(PYzqM$vy;464}U1-eZcv_5%X*^rpsr# zvn}t1;-XVq&sUeT(jA`u1EE+wmiNME?K|`bF|RW;;-{Pf*H4$vV)Cc>*U-JE-#)!b zP?o5zbYb%)_+H^qfk3a{Us(7J=uY#hWXOi=1=FhJ=4L|lPZuHV!CrRWIm>$!O%=C| zuWv0kN^AK!O~uOHclkkYRbMs_)5}EX^$q}hFvE#V#LjvtPNZp0Im=2za7=d?BO{{9e)u2Z5K@_Vw z<$X$t*Jn@#nVFrT=?lfu_pg-@)EB*$p5v$vGWHHH=k_@0`q=2!o^P6)3n_Jap{cqm zRmTOcd0U?M{$+cBrDq}$Ps*Gm^a`g$a_Re*1%<2Uo2q+tyH~XK9LJ5HreZVRP=Qpg z^n+`51-S=QaH7Jhs%mT!30Z*LLFjh$dQ5iy6#8)P5$D542;tr_!R#S1H`1TiWKTFVqab;Dx8*V&aDfxaYON+3PDQKJQA^Ms*W(9Z=l&sa{pPGl$e|StAH` z>1l)eq~`d3oW|HSMS+_*maNaADf|@&Jxy%8!aRwf^fc9de*TLCfe~vtI3UwgV>z?t zh^e&Mt!SH+Bc~3JC&Qxe^qR+!BKiF{{G}v5tDbJpPWmygosN~En=YChViVJacn?_A zdsXuqYf>*+-qV<@engfqt~>KR9PBQvl@NKSrcuu+!PBhFJpGK6%cJz#otgAJZ|e*# zkB_W(>*05|{XmslY*TT;lq|hjtR`YxqqF`MURw9m3o4LwNd^b#>KB{XFDNasP={U4banp`w(PKyOirn-hb3U#hOar+nhY#xQdE2TEhKGwD(tCX}-F@XTJ?E`}-Zqk>KXQ>|ga8Gzlt5|{+*D5}{C;YBfaaH;L$X0Rn9&F?GA#pt(%G#(h zsFVaik*G8k>&o6nZLKQWB0gRIib_gKSfBGI?om>f!Lds;A`%sfvBOmPtXY=sf5*}1 zh+miQk1E5R47eA9ETNX_{yoC<#77JFYxW=}=MZHwN97ytWye!3mHo|vC@4i7e`*z< z*b_e4(!Jf(ygI_Lr_IB%a0ZkwS@>Xa5IcbO0&??%^54IQJp_PaWNJY(O$E7pv68K% z)c;3_(#F5@6vM_+9c#%jbgRYYOCkp$s109Cjm`5mTcaW>%M;~op_uYLiFBf&O>T{< z*x&l$H9@`MYpaTzShkMZQ1NR36lAA2yBm3^z;WnG2;RQ`ajNB*h&KgLN#ftETfL~YpoYoS>P+T6AmJ=v2OAcW??c18NJ zS&;9I+Bez{(s{=e?t!kfAME~^azD`jMDQO$S*#3oYdYbnr5;4Y8}mf6l^8IN%vax+ zbZ$WxV$=U^9&R*`pivK*UJwyIPea90yY5aRw9~QuMqX^(~lMQirpNr>es?~jBEG7s1M@($PrH`RtoZi zQ9YxBN|1w{G&bkxIR0Q(@TPg+tM8%F`mPB|)P)@mZR|X3YHfy#Z^ed}us}EO|Kvvu zd(^9aaSZsJ>fHe~c87(h-PpQv55&lroO`@o>%82L@mQAZWuNkrw7b1sk9oOeBrVa) z{?yA&#A8>Sw=BhL&xo(E27-Fo>2*qhuZks?po51HyXoVy%Cr(W2q?$o%%TJ~HO#8n zTpX6=9f;mI?vuBet|;o_%dBOm$@E%~$q^Z5Rk|0pJ+L-_7%pE6K*Xm;hqk_ohwW0= zsjF|5PxIL>QzYAE5s4TRc2&{#K%b(4H>)jpbxpj@ZaKia7P1o4BX-NjNP!_$Z%^9; zc)Opxu|X6y_`oWgXZ_G%K-|J1nIjrQHH32oZ5nq#pJI?V$KH#4_UHlr5bST!_JH*c zM}R#li5TPZ_g?Jk2E!%xPTNGkHEtmb@jYt8Mf@A;<2B}hOk*uHYI@_VDaP%2#;rxh zmrIS?GCP9Sl*9>HXABP!?bnY8JQjX?gR;TU+i2rog~>lKCj=bZ#&3T}Xq_}r z^vlOfp!onV3CGFxTGJv=*fQ3_+iiU9;&6x3!q+SjEXEoO@6h6}NLP{P8$gz`{3o`A~w&lQ^hs)c`zRQej$ zRd0CL5f;$d2I0x>nllro3F9kVptk&R)@J-IjNT04ur^~YCj)d^dnR<>#@fdkB^GQk z{?_u|@p4>^T^>+9{xI0Fj~7tpz{6dOuJO$MykTa|2AlIbAzEwMUmW_KS)qmkekrDdgv1$@^#%38$?QhB!PsGvF3lfGVo~0 zmmm&!;|%O7{}?6iwHapS)DZl^{Alfo?($IJJh#0pOB%0NEK1tdeW-p`nv1VZn9Hed zAFR&qLmyzU=Iy~?J?9&&vPEMCD+H_@L20VBU4W}T=0`*j|4+sWm!Jx&SLJlLrnhiZaQ%uk}!_wG~sOMaZqEkb*|J_ zE5`VRd9S;q4E57)j@2HF?cAGVH;o=o#`bbj384Mv83EF4$own)*{Fs_Gb?%b4V}%-{+^x9SFbV`;65Sp~Uis zJf_#NahSO36CFXjkCfLbj}JH7##5x+47sxBa76v&FI>!v0Y=)wHz!Q z(zlB5_w8~)VCrNfRte3Em1*H!)j!<7itpplK7*be-Mb5lU(X0ZWI`fSnY)C6E^ z$YlQD#N^p2`1#bFVAyHYC_d$VAD?cAH`$x_J$yoe{{}w!VaD-`e;l9ku!BG!<0xwP zTjg~M31is*0AFH&FIWP1NQ3Ol-ZlBQF;4ks$G8jOYW^YIFHq**pS&%S>Ix3P$;`?C z^>pmx69=G9FNEFfCicTGmS7A4&OZM3zKp?qtmnyIyQ+!p!r3D2uJT>J{9r6(F<&E# zd2e}QinhDcSltTn$VR}wn6VL^ z4K%%!Y@OUi&|O{z_cjUN0sR*~e7BFs%DoNWZ-;-J1V83t4E$yPP58@h!p{QRrS%PT z|2rM9EcWAkj&I1xqO*_B0nlUC=xs1ZGovuy8gf7W{FjFu_~Re)+i2(CgLw``ND9a! zA>j7$U1W&=KD>Xu@xP5DD^+EA*Z<4({i_>3`p&vlXna@PEttnsUAS7|za%f*gn2c4&rO&KnZYuxA;Hz6Zqk`i zkqqQrqV0ZqSnxf1c>DNv16L&4K8Cpa>B*>Y^G_3Z&=Ws#A4^aEc5Dlj*yp_NHR<7G z)b$nZr{<${{6BD1#i$bLRNxz*-+S zItS%(Kb+mi)v!(BU%{4iTM69Y3?{_S@9wElmVJ{8|wFSqfCAC^BMdESyg$F6nIr^3I#)|YLS=+hYa6ZGl9 z|L$79#LoC(*#CCA34KbC=u_e?nHls6X6P~e__j44ZI$R#{B87U%6}K;)iC*uF8lwK zK0(znIp;5sKmQRL75#6%wSLh%mPVcPjwPu7JdMKjQ9c25bbb7=h+m*&ahHg>xfkpD z1QtAB{ZH++mX(g;`s=sjdj0<_u5Yq24STy?kd#`Y+Ok&I|e4^4QZUj_k1iH(b8SeN_@&#=FSA~*fv1J@5u7zZwNcrqPVGTQ?I>~e+40DHg5_W z2#%SY36*GZ9Mg3l1)fteC$}o9Nls;@=Pod-vuy{HlBQ?avZ26P$ffDgnzjKR(^*m+ta5^->Ea-zXXLzb|qbzM)XbMXwBv~Nw64>@OFQDxR4wkFXjf4j`n(| zqt3m?$6*A1L7wxe*Vk_g@5-&Zr`B$R<3_fx_eu9iU4L1ZA9#Z$3O{Japi*lE!GRWe z-myW^B*#03d3&tok2D#NFQgo=sx8hhD&|=Dx+4CSbisp6P=!%p&;8UlLYZ?+xHE2m z0cYO`uVbj2ZB7%%)*4WiDSv^K2@%p}Rt9Ppa=3;$C@hl7d!Mn0&zUUHcJItDfR^pI zCpv3aFc*BJckbV|iwC zKhe(HqyP0rKVKuXw>2SCe*A+j=bxnt@3Har#y6()+76y?H9A~GQ|^cur)?uDcX!RS z+Kz=EI>HYUrF<1!UL>vjuC?hdz6RLSYJAaUd>oG2En?&7=g{Le3PZX<=|MzG!`GY0&Q|H3eaoT+@OkHDn z;ttGg$Co|U$p-^5SQQw0(z#u0XW=cyH+d440I4iTn4SYRBe9y)J@mcdTOWlk)ovmq zfr%T%kylK`o|9oDj0bvy@tN68&fM_!!yooEgbxyQ8AZpWo1;h2MQYLVgt`;`$+^{^ zJAK2}$=j>!rlKy=AK)y&jZpbe(f(w-=zP0pd3Xb!dr9ZBT*vINwCX0Fu|g69Ad1ELD+N(Im&C-4gBK3@|yKm!3$3s#AXuR;4a zi30{KdMf5)yCs}`vLp-#zl^WJX9MI9&PkJx4s7jOaRbMX`PclmN__dR`=Q*~bDJNE zwr7kV%H*CO^h3F;=l`dED1SNl_xw;U9sHm7Lpgr%|N5c8@#cTi59JpdAhi4^ekdzO z^ZZ{w6u7LwT>6jrp~N5j-}6JsU^9)cH8FX{?M22{OO3zP8=Wae7aRqKX{Dgb>{xke2pXjExp!v&9H%{)Uf?6ZZ@?r>d~JjJsNm^&DvWei*1h z;PJfz?8yykoyWhH18c`ZO8~qTZu)M#W?JS6%aL0d!g~m#d}(>`3cw*L z%dW7El}$Iq!EjT@uqcYq@AGxuNt!}Ybnc_~`?&l;-#I^D=XGA^oYy($^?aQt25k;3 zP{y`^SQf!7HBK-~4f`R*_`qI_lfU|qrG|m*9ijKOfEBxMxEWT5(Y6Q0WT!E=TV67F zd>MDLmf2}g>w#Z&`0Zm=bsO8O(p25q4e1xb&N;QabEc4~M!m2sEqz^ejz-E(r8}mV zU{6r2uw2SE!p5S8uGGYY#J!M_?d~~2h!|+mx9F4%>^{EZrUvCocy0ZJTTBPKA;8~FBdgfBj+;_(EXB-XJ21ig zf$O-D$**D_i{p293Ct(yKAv*8tFz;!n-uxojWhTwG)u*{mUhV)*5Rz@edczQJri`m380K>ebT#ft4mq`mtFZ9mZ7 z`MAq3PJ^*?8yZ~Q63No6LVH)af3UZ1Zz5~<+9#?f+)6j$^%6C^eu4(grg2N{c1PL` zyS0WVqyw=1D2~{Elo*2&rb=WJ>&dny)OOq{yD=UVd8K@iJ=lH}8|~GqS#NL}oI7r% zvo2~x)g+Z557z{siJERh8*oHFNWc9d%ngzL0^#fmKTLR3f0YcOoBC^!p4of^%ZH`6 z}#Q! z>{Rl@WaG%42_=|(CDCN3lB22zk8e_K|4lU;7hyY<;b|`aD%q)o#zi$77hyYLqoPI!_$2-1G2F>o@`a>t6LOV!b}>Wd0SXE zWD@F@=&?LIA{#U>M#~0WXF`^loz|c%N6vR6^C)apKx@LraVq=6a0R|3B~w-O!a&Gg z#hwG#m^5CWomt&H3Nj#$^jt4Enk7d|Fekb_QHv|aYYLVHPnzZ&?ml)a{XqHb@Q)|6 z@!EzLe3kLqP(9%r(cPhumGRm_<5gxSweol)jXPLIo``tiHbZ0MEKby%?El6kVv@ojg@6X3^683!A)=feok z(JD7O+eAK_3bgp}G_&MeTHSgFG&)nK>%$j8uk3j8tp+I0a}%Ie1KtF6Sy-&8`WwuO zeOcO+w~WFIdtU8fGd(ToApXf903~ehuLNBI@mnWkd1y;4_VBAODx%{0Gew?COI3o8hJzU?rMlSo@~;V& zZDo!+K|5-?9<+Cqpy%uK&{FE;1XXp~JEofHPhkDt-GBkRP0;Fu5*6v9-6hb5f@=8K z+N`Ph#dN-ITi|HJCFnz*6$}#$lTM)h5={QK3Us5hDuCnjGRx2GUsuhoEVKCduU99J zDtlosxVGfIf+gjJH5L3#BxEum!d{Cc7vN8lB;zqtw_1?VRWvRjOU$)Qc~LxMY3|DE z$1~#Jz=<(5tS3=8pyj9q!O4%>$0_o&d=W+u2w1*oZe{i3*vLHzrT%>k2q!Sb`3g^@ID%+*qgS9J>efQcpKx)`N=n}MS@QO&_GFfVlTA8 zPb<`)?qi%hcglpWqm?Rs6_of1zh@!&d_#5H{jxme^(8O*yv`%OrN=wvLkaCq7q)Nn zwYB;D0bJ_ny>f|ahjyL5CFi*Ca<0^S3G*GgtwWwF$hlozHEplye^6uRR3}KIJ#o9r z^jh`Qvq)Aa4&vq#YV)6H3p50}yjWjkdFtG;ZLPmEB}~`JbEnGkw6<2ECs#VC7G^`> zW;tC>ctDXX*xK%9w-dfo*rk5g}RHYkpiO3aD9WyJb&r zWu9dx9~*ox20{*}o6w$SEmdrk-{9<^9;y^6&=H%> ztI*#Nrhkz9Xr1J54(qL-KzHU(8QsiQHo|xN zKgoA{gz%5mSQWBMRQAII*CZq5Yr;JK;$+0$JfQ5$lnrQjUslg#m&Oz`&PCutNFoRUwB`(tseU_+Kz@)*yw#(%@7@S8}vLX6_cUQ+rBuVBvrB5 z#GbTe=$?Vg8jQKFcTwpwu`H<`duVC(?1U^>UHc$ucIic!(HSQbPVap3GpG?pchhN1h37SMfg@WdgC?qN6X4GP^fkJr?X5ASz&12b9zk%wi<;C|<$MLIs4G;}5` zSkp}czz@%AlqX)Wt}BAl?$MTY&3GR<;g>M=RoE9 zV>k3T>{sCFHjzW&bu~~MebDKU!X-Y$BH1}pD;8`m9dxtvrOz;roxCwLkJ(|{3E_;K zFFpFX&SSxf$XqbwY;E{?%nsYm!F3z!U(`Id1d7d7&Jr@e61MFO&j~PxD<14x3^V0i z@YBs>_9+qCKXD#|lELON(v(7TZ`igoB9BAn-jk!lww+;`VGVLDn^H$a->ZGWPpm=nY(Oxe6;V4paqN#K~6j8Rc9Yi?Zhb zURCzX|9&xqy9mqC=e>pxY0H)C3>I<1U3{7~JoC>NL%54D&0pR%y!;qsCwdL9XkVm> z-qZYZYxu`cEVPCf{LH-Rk@%%!q8CIu>6))BlgbL8{TOa@m8Dde%@&m&HC}*I4$n*l z5gSjRV&mC4CM$IwRw1(S#GW6vTi7q+bnZz#_OMT4t(oT>hfOq@e16MSFryG#>4GM6 z>A{OQo=CBciHCgNf$iaw({*}L#zDe?OZIQ7RIh%EEI#2Pa)`yJeHJ?y+X|CU7=FU$ zb7`WUY75zB#=87ZVp&6BWb;Xf5?g8Wx)n~-uy3Usrkyx8bEW{Xhpo7GZtsI|sC{QX zuB9jswzD0UU*K?MhxQPyt*+<(jTJ`QVqo|QsnW*w#gW=__9Fpp@weLXA|oAxheqcDz zrd4^kj>bPDmtc9eYai?nbM&wOJ|N}Pq>p-&lV9UZ z-nAV)UDaQG#%(w#x=~V&zS%uCsw@)yDfAzweo+Y}L zL>1NrD^5EuSR<@DA2{`uDPR>|;C$1tXAjvqL^JBFV-K~jlgjxfC$sjQwd`rzarVE@ zNmcD0R$5ei_dMCT`WFT@*GS*o?2@?C33-E ztD_52>_3{R9BrhW;6dp6TLfP@fOJ)N-MQ|=P);b;d{d3D@_~ent|ur z+#H zLC9LW){1-hinT&k4Go-%nr1O|upffNf|CswN^2fMku_PEu=JHMfVDrNUa&tAyM$0b zpa+NKdy(~D{e@E9_t6P$O=Uq6fy982Swqa1ejDQdy*c|dqj9DOGMkND)ohhZds$P7 zrHW{v14Cny+0(4QP&(~CrJXg|9ds7?Gc*mZY;)UV+r4JAEXDn8$+F9fO~($*Zmfx3 z1E$W3np0xjT7evcL|S? z;)T*j@1cA@Mnm^$4!HLmK`74mWi1=plwQW7|D;f*8 z{}QCLvGAJ^-~N)*`KF=krtHvMba{Kq(4Mce_9%_O6^?eExT3W4Ey=R2-p^IF<5V2l z!`24TS+O)c?z#A-n5T9@hNA|3roz4QwHS_Qb~(70Hb^xayG{Rop>*~XROifeLeLhl zX=L6aFpwB22Am6LG>@4~mTz3mk70KXqw~eYLdQ?gkHp2x$uC8Dc_H$*VstI341k3T z@0eFX2KBJQxQ6*;0p^pNz{c==^3ij{%qPnXT@}k14>1$lABY;_G*mOc3Hi?^Ix_9c zRVwi*Fw8U)>rvUMY&J*UPD)UbIj=t zQ-sX5m{A9rdrFL5TLH&95e^?DLYi(ve_BEXY~^{nXbfLDk8mA+4wHY%qp=;9C%Y{_ zHR0m!CwnxQ5%j$#v8#7+N;iQPej>wkJ*cH-W>M*ZgHA=b4BzO0kVr(=9yM-1M&;Keid>Gp4ZP7zX##DuiDbI+YQzg(QswGg!} z29NL49em-G=!PRDFby*gc;+>@pK2C$NFeM_JlGTb{eTtvl02nGaDzj;Wy@|io_Hn> zm8}&ptKl!EW6jzu6cRljccT&IlR(c%>MZLGZvYd63+V0el?Wv(s1;X+E=MMg`9r4_fS zE;V&)RmC~5>!)DPcC1@6!jYBf2<780;07LL)_M@SPwP6j59x7*&+r4+5q{t)mcX|I z7}kw8Rn^{)s*ZkSwLC=tf`8JF<;drT;0N^2Md1e$o*NcFkp0}36hH8%eOJQ|T-x{f z;|Kn_?|<HQcepMW2@E1c$k@dMY6ADF!FONJj{`qahPDEz>~)USf25%__W zK9yk8^{&M1iRI(usq;{9VSOBbI+;FvZ}i8|2UR|@ z{>^Z;PRep*oae+i@7Zw;-IgbnDvN$YZ#!lF0+!!T*}l(+71JPpR!oCGe!chE2&UojXYG|COhfr=u@Owex$l9s5llnVA7Z~KOv79V%4f$k zG_C%hi)p}tEx#!^i|J!gsD|5sYPhit_nYwZa_yLg5C1t#L+1Cs1egZ-nlTMBG+1C7 zF!JG_(t`6=@=y7hFb&(kP)tM9onLlLgUrs4qcIIX$IfFYroj$>me+=9Xeth48sK9k z8q?6i?%_T+OapwZ47HEmJ-mH1K^m&yVg*Mkiq*Pu2~ID=sIGR`3_q5aV;UYt^2jyl zr?L?XV;U;L^nQYk7%&arR&0E5&wOo|hI06^WS9onQiA2o$_JQ+*+VGKk2Xh-2YjRo zW`qnsGORgA*!Skt2%kTJZonE~I!||mfHj2ahBwXltMmC}=9drG5VC;``8yl_UcWpo zjBAJjHVnBh{0zW`kj)Q(4be7`ABStuxIPW8L34FnL-~kdaSil@<1n~}>Cu7|2V4W} zQLho!04&^PcAhQ6;*iK@J{r_e(lsooLAI%YYM9XlJ?GUL1=aA4E(CVi)P!sJIz?QG zYnVd8SHU&pQQT#?hN)fAxQ3g%hQ~GJMWu>!<$&PJaSih4fNPL1!!^9H2lUDcuE95m zYw!`S;T_-_GPCFTpir-V!^AYv@$y9mF;Kjsn8ChBx*I(YS^`?!loq z+E7Ng2Kz9u2HXn~u3-z65yCYvh5*;_JcSL#HT;qVS*{t^@QjkatMslxTtoAo5U$}V zij3eIeBoe5$`^-gkOy%M@@K#`oSYmBS^9i&4NHkpAB$@!r4-kUYmoo7xQ6k2zRb7= zIgD$NLqG%h8gUKs$Ke`e1=k=);~J=fD{&370(%JI8stG-10t@#HOSY5Yv{z~%^0@vWGMPVMhwwS#8ZtM3@iyzs88yQ-aIliR7qC{SfKa=24Au~YCN)Ht8e;pJ zx&Di$U%@rBFG)S;8$N%#$fnZ&IoO7ibD}T;VSC^q`<(w**oOB$6}F+g zAd2oU4YpzY?xEO*{p6Qg!D}cu6H)O?9ocgZo!=*YNwE!bG`6AT7trMtY{Mv+87bHX zIfQMPznn#~^CrSJ#B)PuF?Pr_IEz(XZ5G2jn%3|nHWb)k9~Rg!?JDq_Pr_&YKfyIL zjrmvO8a{ar1NM{7ZLS&DP_Fx*g=;7uVD{Z#D6U}w9BmEa8m?W`o8J92xP~F3u8Hyy z^GUJ}TtgI`4<|b09A|g{PtXlDl))P8Ay|XvI|{5J$Y2eGX$|9J!tY5$=Cq;sn3MnW za1Bkr|DS_vIQifIQe49);dwqIu0h^7Jg!0h65<+GY`~HR6Tr^}*C5Vdg@nq_E8QQJ8&cRAEDK4JTkS z9))WNV+s^p11scL3a+8)ER{2#7==89Yj{e*H9Yvtu)1cE45`sf#X?50H~GD)blNJgP1>J$IksVk`cq&V7#kS2A@jGYQr*KS zG78(!a8+yr%cq!+9#)msL}MGyK0S;rvc?d>Ho&&s_1C9|YbU{8hPLzM%c|0@Prx>m zTpPB5C}!A(rl6|y(MoD))wN(7SUzSj{k*F5!)RbjLG@Yzfl^%@7HT4(KN4APRh2JOiz`_#CG2qBqH^ygJ^4Nr7n;{Qyz2C9JM|54DFer>o0MiUL1lMi8h z8|6VmXAUn2Vc58jNwh}|~QMlcS~QvR^1>=kyp z&xB*?1%>3Q^mCL~4#6LSyFLo`hT#wN?tO`RaZ;Isb8Vqqed^Om=PmoVLowyIUd+)c zZ!8pY>*62dtiChjEp98`g}`SW=9!MAG&Ze#PVDBQ*qyN1PW9N6_G#ce#W>uR`- zpYQtoaT_(nh914Iz2C>HM0Mvl(|Eub=B#y(BTvh*oyi(bakShS>mFC5I9Oiwu3N{C zsd_|(r>aGtnCMZj^|vB@-QM<=39%w&sfN$f8ZowY957Bj%wee)_V&U5_*7Mt2bX-R z^l`y&eX8^~!&#!JW)1!#-+ef@lm@Ef^nCr-a^JZln4uDrL~VsO*`L(ybMpFQjyLDK z-U_GrU);vE<2HWN{srPTRN}ZD+`br|tL?+HJ~R`)agNvK=-lIYtvgx4s>KyKyjI5( zc@ED4$K&t-y27!sXn$gDw_w$ku2k(HJjBU^=kOv?7{B2O<2O8n-^gias|E!!{Dw!t zZvZ#<1af9BdXMlMEU$v!*j)7iP##Jdz;CcJ#!?v!zu_5*-|&R-8%pI0euHI6q%23q z)twl({_HrZ8*g`^D9aNe{Kj6j?AUDo6U>Qe4OZ|Qf}@_-hM%OUlN{>>M3 z?@UPK-o=*KrL;b=t=&Bvu-Xo=d)=IEawSE#z{NS3jGG*=IATVON4YlAG7O%r`p_rB zSxm#-cxJ*|b^Cy7$U76nBZLomX9CW%H))DPm<)3WlkxMYcP6~-or#}A_UN4nuY@8( z?@at$d1oSDd1oTn8^_+6$fZt+dSSv#fO#)on9wAs=4#1|B^0f^Frh?O(PPI!@Jls; z-d)gqWgzst^JQ;L49D;tn#{WhkMUR>y)p5h^>~7wJxYNC5zJ9hPdwjp%}+dw^u%-I zjR|_jmi&3*QRphNuOI%MXN=M^OfE;&LWxs2%8{4AY4#^$LzoMCtNKpd#|~mHibvX% zcO-^y)5*C5#X4KsV{!D3#D7*d5yXLIY5phQwJlDth2D`U(Pp!EBtp;hM!zW>@kt++ zt@FiNo3RSdNGy%D1Ij>#Z0t@@@y}ZXWP$_CmWOXkX2C#FgDSX)HH451NFV(Ekauja z{f+tJ1h^M2DR$1x&@3gtZHCah0ZQKt|Hg0;a13{{`|z9yj-e=wV;G+1lamLEA~IFY zWQuZ-OhrG7{Hf_KC4Eg$F@0>>y zeC_WzhZxN;42;-l0ihkUGGB!!pJHvnS0`??s_egwHJg{z#_Fwjpkcx>-l+K=<~XcH z$_NN%N7@qdNqrUmrw_mZ9W$z#^$uqWz` zbRWK?Hrak3tDn=X98&cS*HpFrk#MCN;l`vO)Qcf?!nB3P5jYaU)Btc*3iM|Pj|>~oZi*;5oECCJ#b)j1OwksZq*W4m+Wr0lSa&5>@Z z>-eq&$3+J|@+2rk#;YiuxEnt2lynfVIf9ZdO2F>cRej^)&~>5en%9N$zg<$+1+w*w zS{LkFKi73Z@ne1E9947qxMvH4;>X&)IC@=R^E|T^8GczHyVXBlDoRrYO9$5l<^wZ& zy7~@WXaxw*9v{1oW$p{pJ-CcvkVf9id^|}b z6;E|_ZAXI~jrOicTS9AcrDEe5nF)7jqP9`x?UyRi2h7KRXicshi6iLG)|g6CewG#6 zBaNXo`Kr=Vsa0CRbWy?5uyLjOke1i)(75upV{gjGEvX5V*8CuNUadmdAOA1guQu>; zRsE!nv-i(JtM;dupBh8+Q`KWb=BFT*!kZ)ZIut-kSAQzv#r)G=#Y zZiMZTB2RdOFk+_06j1V)KU8nkR1B~C4L?6s-G#+G#vbKoUwNXd2=mjDs|+8SpB`uP z)18B39qDHc>AKG{KULi_Jnh5IPtdYq$4S_WsQC#%=c|qRayOfw#t+hr`7CV+&7Wm{ z8ma|eRpN>uRdC@L{KqxVQRP2mW4-Dh@J}7lSvBtre75lsnou?5pPCK!Pnx5uehvTB z&u_!HhI$Gm6??R6m-VIVHb^2S@2A$vw}R*O)hn@k>*p7do8n6i*d=98LYo(rx1tWcJd7 zG<%wa737#YcZ*rt`6GIs!eoBjT(@U+ueaOMz^4YDSe^2LA>3W{9%3izfrXA}x{m3gbVblN5IO069ok0Cc(rLUN-MMN1~QqNMox`Ox2 z${$^ZA$e3?^g?b?@f$+p8&w6+b#NogyZMT6?Ck*!|aN6WBoc|`MJJ= zEH~{c(>p0Q!};`+gTAPXWB=kU>|Fsz)U)!rDq!a=KKQ0{Rs;@dvHp#niu_1VFEdr* zFP&Q{-mYW!ajtc3v|`dquZq-);A#p+rM*Q|YUk41bvgQ~V|as!+Adid3qnq(;c=RA zz&wsMe~r0Rdx`m+hNWZ!IEHiS<;&j6XAY-vBu7EZmu<=yj;Tond4q0b5%vZv7B(0Q zEDi8ai7==&IF?SaO~tvK!Y-*ea{TmI9KA)=fG17)xS9%->b&g^ltBA-?X+60X(n$q zle1{#$2*E#*mv}n(qim{H(B>57Km^hooB~+0{M*|tN4Xg42#&AdpC~Q%fEN2v}O`q zM&nezy~7EI(RfSu^#|IHSn4}0_3-HmztJP)Bjh4>ui`g)EcuO=mg|tN*{%5Gq|yT2 z&Zgy;U?hSAe5b%p?;&}hRW~ig7Tb-pHF;XYCb6sAQr`*Z;HJG^o==4X>hwKrYipa= zlBp&dr`%P*TeyF?+G2$odBRbj z`Rz-kv&}Nu6xFxM@Rcg#u^~%6Tt%jjkdAQst!Urcw$43&$>{(ayPi8R;?1#2Sx6Xm z^B5|w{7&%S1iESj*2V_s!oaC9oZX^ghy194h#7h8dgg+{#(X$Lv(*2QDH79JR}cn>2cV%f(eqNn_umr5H93Y*A* zWj$sM!GhK`c#X?QA64#i+Sp6t(f2t)<~$S$lpe%=PRXUx4@ZPcy5WuMz~+MDJ8(ub z8zXZwImYbM1mX8nLijQ`#r)#}c-Ku(eES;rzzOkMvDrO_S!XcXUTm6qsr2_+VimbC z-rv(oMt&m>Xqf}!&aQiNURY*?UsdM5czsm9%)(14y7N`Zup5CR%rHfF$D?(c+dqhw zyMsODU%!;H{30e|sA!kjiqKwZu+*1M-g+rz{H3d2{@v8drX#w|ySH+bJ^lkQ5+)(W zqaU=`cVDtvHY;`&gZ3a_p$voeAes4>{uTBh*Rfin>_M6)qMj&wkl(#~b$gHtevqip ztyXtzh&RjTqifaDXuK2dj;pJWg;z13v#w@?JC^5yiK;_tS0YunvFoCkB^Q=ihY|(s9cWm&&0BVEH46IW4 z%saK5Mz4BJJ!k*E7}R|7{ejA)$8@k`;aqP(H57=UlralcI)4xrz!4*kvZ(^emd-9O-Wx(ey0d2>wl^a68_`PuEcFEqVcdt~d) z$`hb{M0?lT?wtoW>yh+8Z1;A~xn|-)u_v~s)*TBLI=#4l9cafprhCjey07UKo`^*& zXWY$hupjnwRmCd(!^f+W%AQZIxuxjEWKT@Z)*ifdUpJw<{#mXn1ETLu@~UgrF4+*M zlM1_EHVQq)*J9t+n7l7_a*|K|ko%kB$n#0gK)<}mo&5&~dPdB7ClEO2tPPwAfLLc>BBEg$aYi(|M;uVkc}>*S=Hsmg z{lmCIY?&PQ$Zd2zE9z2uF4;dC*bzq@&px-YO`}jWr)tim9kD!b@>V~s&~sCtM%2A> zyx-a0e|&G?;9d~*$}8vE0~22n1FXLGcB-!(_37qR%xNpA_Ib3mB~^H+R{t=L6RX2@ zdbBCfN>p9LYrv!ve?XGE3%L(_~+w>%A8AgIL(=>1{i_$ItT} zzQ0#(BQGrOH1~M7&ONPDx9Me18gEpk7O2EVe<3VpZbU1Zg8gD4pCEd?+ONeOJ*m3u zX5N@PvCyr_bBxB781id`mVtn%u72JoS;q0_3nnEAeK+JzFLWmqOfvV5)|};n=fE44 z%j=gcS?&bQhwvb7kSwF!uY3RDf4%b`-O78U50Iq1T0?~x-S9k@Ra@y=uVk^*`m5gJ zdS37Qhre*nKi*b@uFPrYTIc6J{$G_zo800*8ms*ne$LuLpJ9t!LD6VDnYtI9%4%f}}^)+4R1Ad2Ml>b!WzS#hz?9!K8;%2%yU zKZ#FGuaq2PrsR)xsr~-fovYjFiP+A6xGXGNnhHsviH(_85ztq;8~IqgrRpqLf-Y*@ z_%8Z1&yh4uM7Y*{7$?-Z&L!@{yf$|x6tyJJe%`~6sn7prGcM*2eB|EG$BFy1iVazd zGP7(+RB9rA=Z}S-b>3)KocAO)Uw6wRyVk`&;({9wQsojKUG3h5MbqVR@kXcn4m(s* zEsCP3Rp;Hi+%7&L-A3aC9ZbJcB}cNSkk|XYd7A71KwcTbwXka~Gz ze~zyA!^24=v5h+$dB1C-BFOwH(bK9GUCxR!_JZxaio}$v$)`qhRc6)tJuSJ;$#r78 z$9Vd6QG5O$x%XK38>*LPG^+e=M^8;(gZF_QBlnGRJ;lm$cpg}?Wb)g(_6z0}zL@#% zw_tsYueiP^u5Zknjt?Gi`px?iT+Nj9wB!5(riXns=JqGdOB|N93rkBc9@4&+aQrFr znO~Y+4V0!T1;V5fU-$gO?eIRp-;fd$xBh1>x$~Cbg%^-ST}DDb?^_nvdP=|k4=t^Z zJrD3&Qr_dsNB1Q3fiEep8j6%GDW-?lY39whto4@0U8*gX9hwdYOiac+>p6U|#qa2R zfahI}l=Nr~C%U%xiMvi2xvHP4DC+Ra7R0WkSp6N9yQmt|!*r)#>AE`xm)iJucMK0p zN!R0)rpIx_67TvC`aHawaz00&tMYvq1?Oc)%2~&CvLo%R!&F)kd-$nv^t(BY3(i-cwV2og)!mk~Gkv)`H23c|yB?zGjcsOvI;&LczuBU%0$0e# zkzh$&z9Qwk{x$usWi5wvyv}7GELtH8M07Z=TgTCe5ge(ma!LT2T=&oidr#WYfi^hk zr({VJEUD6gT!pw+2N;LAIaow|(W|jhyCd zs>4|=vL&U%aveC5Nihl)jKSs>OkS?9P$+SCou;a9AXG@&VA3$bX|Aahc%ROdPk$ry zmIZTNlPN^7q_sNAiZMHw#OnJ0bNGhAn*V(;N9P0g-)nY_rvUJ|>RTbQcbQtiGpmxW z{SRh8V+AAQ=wG?%3F|c2dumkmY3J_K23Mo^SquK&EI888vUi%p_3wEA$4;Vpqd1y> zGuIn=+&aZ=aJ|utNvBx52z`}&RIA1~*Vc(E^dj8yIGqpn_J}Ui3V4}!RgnRYN_4$t zJ3J18v?F%jf0FIDJy)WK%ihg?>`fBtvx>s~P?D;m;;u^&`fwEd%d)B-DNXbs6jq|%Y z*Ru+KO97<~^F#;S1KSJK`nKK=O$({%a{f3# z8c_Nn6hXBYYl_r=e9k!%8+*7Wbw^~`@vCtG6i^W=B>TOFPAm5Fy&zj%0_7;qmJ z9iGGP`I}054)4=y(6*k#p(IVt{#GRJ@X>PIRNgOGl3E=p{%`El`3nzeDeQ;+=-Sr> z3*YKEWxGlFHsB$0%V;dtuP(x0=bEP4KCzLWe*mF^Pc1k`q8vVL(<0RC5Q|~P<}W7N z?&wz*XtA*dFNV+Dt`iO(2M3!M8N^2U;GeN{x6N#;XcINWvaI4Hwn%vzg`V`9nKc!} zrsm`kMEN0VYSWKEyV%%lOzN;`(0XSNvyuE^Kjv3VuC+a#4%tFVT+_P#jSc9dyJOtD zg0QBi68I6_J{{4Nk2WW^&ez|u$w*8+;!`OR{BdG2}8Y*Z7q(FTA_V{PT&g+zJwD#YpgT_|~k?oNGuKDUERnM%Kci-51yf;Zc z$h&@63MmF9>HApz@AluW^Y79N*#Dy244XNpGuFL}@xpt!$v)8P%ERH6C2k+t)}Q=A ze`_PPgK_la4^fkUTDK+9bGY}f{`pIO!6fXFe$QdwK{H6yNRFffhlxr{zqr4LWIEtC zi@QwAoNeabgVP&YM*lnYh#k+OLs{9_r}8(7`=QVIcj3Q}Dx1?MFy^OA4q|@O(*90I zE3vHn`~9BAQ)#iiUGYm3sF-BYmDe*duS69rfGjkpCnIn}{B#d4-*I; zFr8c&ZX1TxskD^0ejoov5@9)Lj)E90gct-9RKw3vw7w%Ifu=4Q545M9qdo%VClmTN zt*L!!)^#8sD=~#=`>7@P%GCCMEshAqcX>R9Ewpf)cbZl+Mf};yW!fqpyoCO4X=-vc zHPxot`HMj)EKt)ea#O7_7*C)QasMeBr->V&q0Xn-liB{ZFtn#0{9BP6H?%-+{@zeKlkkZh;)YhTwRp}?~Q>hsGD{?zH(X%h+P%P5Y(LyK; zHj7rD8ezWitHXIju1^wd$)t@J1o{WlPnD=F2g-VRmzckULvQ5GQHA3jSg3@;Q6hKY z0r518+!xQ{HqKVnPP3{TIpiY75gHt~>lUOn3M%qd$Kr{Z`Jx z+5vWQZiCzE&T|*J7r4v9W#kpvBJI4q?E*LI;nVRV2QUygWXm(#%Iy~iY*x)jAP~F} ztlyBEHhE6x=04csy+yX>Zch_+D}263SBRB|;u*(lKSb~m>o-K*p?DNeoq$}PKL6=9 zueT3d?$f5$*3%7QUh#C|8!MBaiacKJ_GwXt;rJU_H|e5FQ^JTe(P7Mefgcn;|WaCSb36lG~Se~aUbIL#ilG$S$(Q0-VyQy!^^#*PAncQ zBO>Qn<8)b*&y|kD2(Vum5M$h2)dykOAz8jm76fZvZ)IND#JsY+v1g_l(U*#xS*@Sr(ss#YtvXM00&M?Rn6m?{eV(#)97fa! z{*|5L?ydwBxDtLdtqI}r6^h^EUnz>!U2#L=Fn(6J1-BgLO2xPN$jtXW!MqB zsaUMo1o!zZ`6H43A%w{lTC{2GnVJn92`W!T4UB<1N>osE?e8gLXdQR-T~6YXWG+!P zHKX@nMU0xy#@Es07|e}4mlZSO=sjGK*1&Txl&pv~5dlYMswFP5^pQ)F)==bAtGpRm z30z%?Ij=aWsMr9Gqj1os(CSh-pU!apIKx?yv4eAt%ecR;-)s-G=LftK3ksLyPF(V~ z<~8n+_A&je4E-U^W88h(9R0eH`$pW5HTt#q$M{1dUem7A?;G)&#$0B8CC9wvMBsQ` zf8gPOx6t{kjQiIwIsW(9gYS6fAD?eo;Zwt(`HGzb_sZyaRj00R;`z0+woKfobwAMR zIHWl~3i}1|0nhxp7XyLze&@F{-v1B1r|?m+YmZaRaQ=ISsRBd>|308P#B%JT93u8< zRe0Mu(w@*fT{-WSe&-7rXd7cN8+S~W5|`ja?6(=i#wC4BoZk{P!-?(UecC7347`!&)eDQRN&#gX8v#FdO_ zwi@|HAs&U@O@)wx0sPxcu{Usm^VcRhW6UEjnv zAMiR3%m>RrBqaaz10F}ueA7bkXjW#x^c*6Vdo_OrcSL$9U&QF3%min*I8_K zx-+uIXV4P%Uh%TWDT@z3*QKz_Y8ShN`*hO{A~$x?n;@d*wfV zTzapZfKxy!Uu^V=xlqY9M_8YdTs=;rh4K0RpJxeDTy<;V(bcNlo1M-(GV**X!4NyG z$=R9FvMnxm5gtLzXdYv!+hTFH9Tq#LwY7*^lNN-<<*&DvB2j(|d6 zt645qnM4fZg&7rbZ_$biP>MGfDOp%n+;3U3M3&6{AuX~jfF6;K%J(-$t7K5;I{WK8 zZ|&x^1-^3yp2Cl&`+N_gu!8mnk8q<;aEbYOzx#QCQ>?3*Byfo}t5J%;CDp73e+15e zi|H9m4kDQXZHEKi!-2hreg48aPhop!q5W;pP*D7zx%s}pn!`@9us6NH`QYz=E!?iE zLXU)H$h1Y`v?!(}^+vrH1a#){+LSj{f^VRDL%OhY0M`q3;*mr8_Dz8!2ClBI|94Zz z$VFPP#aWZ27ZHbR5BvKlP~ufiuTbZ)NOM)DUd_9qI!L6V$3L&*pAceIy(>-Q&6YwY za6M0=;`f$aFZ8rxk@T%Ej9)%+${t;V)SaLT;KB;~p|)Tg&|<%2>EeW*1DbOKFtb4F z>b<`*&D4OcswsQ=8l-@jWJk?c)y2c>BXooX>$o`?<88fD^PW$f7S@NQm{DRe=FF$f z55LKEWRMO|B(gc3;wp!Y3%BtxZuSglCmYmb&xr9IF)Gmp%|xqevzjJF^d})^)DG_9 z%A}r2`(gq;^x2L7Jl%X+ZkyWYHRFdj9AX|4;)XJ;$R*R} z7l~B5EkD$bkQSP9UL$JxN2$n_b<%H-d&6a}96l(oGK(U#vQJ$Xsf`NuXdnLIef*Vd zDn88_j*==>EUucxS#4Qn^MfFO+Hc6ULc>&eN?H}d*vf__KYE`U;i-00p2yJpfd^eW zN;+_+PAn68Fm#tV&9PLBQf9#wWrq7AoQCmY0GcoaZhXAc`7@!%No;QK-#wXg*7bYicr8raq541Fp~t+BbbEBc4~c)` zip|)$Y+5NM_*1294o6dUHgm2uG1r=FK3ibUWK$v1&68G`*Q}VgWd*L=b01%Uip?_v z7_qqv15h?hd|E;4hIvn{u4|xS%p1(MT&3+cPKhn$Mi{^6GUe5m7Kzb!_txo0eQY~*#!9b#KAl=UwQQXl1UP$k>9{D|e{kG`FepKnR%j2%~cUbPP zmR^kUjIQnR8f%Za6GW?CbYp!$LoW=t4R*}#Z0ur%XJc~J*(v#7$^CJEoz3UrR-IOj zhSuSN?BPQ-v#|hX1nk6(%?NNd8)F8}G z3T>382WgHCrn$V$VZMdW#lm%le@niBm#yi24~a2eZVt7GXd_q&J3TQYw+^&OWah{ z{$oq7=o7UnQ-P|j(b<;B80**vQRi;q zazE*>)dvzlWB9Mx*}v+*tzvF&KPCYj4AgAkx%w&B*J|j|dUqUe6K)&98<&mATNsj~ ztU8NDQt1bTKvSsgV99lqyr`HlFr40w^r%bV;-(n&I9V@(nO$B{9Mk9!PCCqXn_#Z&{#R0VO`5={dO-n=P`&e)v52n!aFpd2%Jjd)J`#} zL0t7gT^)8oxwB_TF)HXSPH$417wXB4lTI09Y3TLO+JML8r8t!wr$pAcZ(JC^GS)i48I)lbWjeyz49{FKkGvnFmj>mSW?)oH1o?>pOa z>uZwaozOjU(G@7cd_GD~uv-65gl_Nk9;-OI+IyWRPt2(HCVGm5jF@V#R^pO{m_)3^ zxLoNE@z;7p&F2e6kB8+#EbClBcNT}jI||Uo!i=GFCi*Sq73JberLsp>_9B#-rNa@= zBDxiSTT`3gxq>D%@NzJ&a`-rb%aeBgDK|G3a~r8Sv>(X>VMRr83yN@p9;Pk4PKq6Q zi@HPQ3NaIwEQy^rFZOsv(p?KzdFNG_QG~O?TsC7u$s{BKT?Jxc(Tp5n#!UszbJzk4 zGrn3hgVZF7gAkt(ilz%?iNXS_P-G9+#h(`z#0o8|Uhb$7_z${wuH-*_sD_t{eCWjj~-B0TC zAw*pswn46Wyt^dEBu4H4CU|KC)NimzTh#T*w@6#K`s9C;wy5eKA5~ic70~9f^)5@V zbjQ=?LDV|eOwO{!3%>i<02Ja>Ldi^w$dq4P6tyXp7d^I=RD2;-h3y!Ny-fp5Q%djx zqE}u(bT+luG1#wRo{1bkRp>ML&(|MJ zmTacFy5x#%y=1(;ZvCI@yiRG+^^(yrWtr7~9*!#9CfR1HKR8CxPXCWbp%tgsKued3 zr%T2HX_3u;o=@=kk}Jlm`BuSrJ+GTmsdGLP%P;Tt#kOzX;0(mV4qneM4|Hwl@^&<; z+B*|@zWv~a4q4R>vsUh)*ExQ}_-rB!w06SJgtlW1 zI%jrlowbfnXsL{q^c!#WHdxu*3qF*{#+miHj1J>%a?`6VrZns!jhNdmaSOd>CQw%fjlH@#)!ADE(vajaHDc#Ym<5X)-&gg!9;Ep&yTeH*h+sBPgnjeb%) zuJ(5wH(^hDQN1y>I@>tGn{I{S`h?c>yn3T{f*UKHN6jA@duICL>4tjad()RqNBrL> zyh#zK(}f_Gbt*focU!wBE3|hfhj^0SomDyd)Jj8oQC6`&eL>bD4bN5NtJCwaA3c(_ z@;rWt@1~!qQqHf|ap@&IXc?8hm|p-2(x>tuM6FChQEi~vlI~q?_nF6qWkHA;;dD+m zL%A(;kDgk_>X)r$m_Ur(2QTtteQP%a_N{Km7XDpmy$?ceYV&zR=_3$l!HJ+B7gnUj z@P^(>kTjliW}722l72R8<(T)xj0tj9{)qRAdE6E+njloz)Dxt!XC`!H6`#ki%6EAK zTEXv`4#NLZJvDK{i7fF?ymx|7@9CRf7VIBDBLA=WZh9t!c%~Pe)vO?Oviufxez@iOlnjic#lqM zbRuRCb_WF^`uWs#+!?PrU8pw#q_hNI91LM{m1A6#x9S7o$Z_e|aV*HtkQ1D(8+en@ zecXxTit&Ry;1^__P6ggwn8xQkN}mc;%oxHAovk>=)^nDP6{Q_bYAE?FC97W z^d>q#D1grVCTV|!JLt}X$3nF{a837ENvS#E2a9L^tOdEgI;<;K>lJ z^kMZ(67%STv5L~t+;@MdUu$vbFO=aI!qIj^)@c*<&(s@m{DZD^S5F;*OKGEq?}nrj zx6y^3^Z3>CE?-9DjZdA2r4eHBM)gz!Z_PUY2MK7#?t0I!`4v?-kLObh5Rc!_^HLSA zdX;+QtVL<>Ri+_*zJV9Aibvo)CCdoKq{4>Dlx7)rSd``I=Uc0%j+($_RsI!`D^FUt zlJ7r$4^16?I0LgLLA_;V=ET%46>N`Yg$6ncvv$8?u6vMgsP9UZ*j0nIr}Y=hMkv<_ ztIoc+jI;I}U~Sh=g1!4RNl3sD!MMi7LyG@Wh)2>3=cnaiyQiNtMH+$4fhrb5e-}GH z*VA52hWQjjKp?{BBhxcshAK#R5z3yVcM%l??`%%V=Se7CJYZMmDUYFZv>#`8?cP9j z7bhJU=*)1o1w?DR7&vNE>$`j}m+jDZY&qS&SI~PKHg$>hH;=v3%e73tj@SEU=GHCD zb>6sQ-b^!ZELgh%(?CR!QLl&d8`g2qlQ0)e6>ZM!u&s*0W38Dq)lfqU57H@n9V^GZ z;!T<-LM1ojJko4qLzd#F2{+1;4!1Cf#}!QVH0!N$)Tslnt2N5cUu?5NIb}N>p$L;n zw@%e@@M7S_!T|KNIVughoyz(>OnBIDYg{twDxawpQKNR##uh)IM|&E;Uq7WYkbc z9bF8qqZ;CkYYoO_)MIS~I0=8lziFPY_1J9I?tvlat>O4|Ou$56c&&?^t%v5taJtL? z@@UcsO7R_SSz!-HWZ5+Qo~(Q| zU*csJGq@M?bAg#jNFd&Y|K#!4JL{eac$1uKZpP2niB5JXE>3jTK4`+_p5UB`-38+# z=K0)Y%IB7M{&!$f5)^yv+YsP(vv`rCNYS|_nVmc-eI3cOYTEJ3vJlPTJi?GXn{s&WII}`m=-ucVRe*Y-?cO_jzC4~yCr0ObHeJKC_cqok$ zg7f@g(wqsWA(?R~eDa*Gt6SrZp>j^wdGGON(3h6NM&nDeVPvlH66+*eB0f!C?+o^1 zG=<;6J`MD_6rY|R`dUq2ef0Hx`l72f**2!GZYDljCwezf;6#jf@6-5_5c7DPG{c5( z!8XSE<2YePwud@h$R6VqzvcdQ=<_Gvaz7s$1CYfugnEqgB5hRIj{Q-n7eal*Q{PCo z>sNh%zjEdE^sDkKEG(Cw)6No0mAOj5svqF`hNZ}DrO_FZH9n2btZNz((~+Hrn_$cg z(9!TySeEyl)j*5jv(v~=2<)Z6Jw!^Uan-AVy;NA}CqH*+?6cum;rYb8=P64^_Vsk4 zSkp+gwZRC%-0n|oZzI~|Wkk5J>?@)6=g#!4ioR$1~@GQb^&Pwj9I>W+@Su+m>$t-}z9`-w^2C8dHw+3s6dqZil zhN{e-lFw%?EzEj8FKhHHB8X~)h5d@^4QEPD;FO@^w$oy)yBo)F6-|Hi#QKxHl4o42 z?Cro}S23m+cl$!oKViush!S1R79&YinOSu(-w&Dz4uj^am!o^ejNkH`@yta&iFGP zYaAY&?iWe6Am_0BuOcw3QLJTCz$Ht~@F&GR=z{dNQD&YC1BpzFD3GX822&h;FXGS# zd-T?J!X5?t_*XjB=!Z-aiEo6#IJ!m-#LOQLBwuIGghWvXbtfeKKhod9%VK`)i$?eg z-2n=S-myQ7lcm0T!&c*li7$laI$Z;pIZ_aH`7}_h95NC&VJQ$b6pHS>Zm*&#&uuFL z$^(h)7`TAeI+E4I-!_UPQa!xq&#3d&svY`Cls~mNyceV~FyuzJ3UGk$DawWIp z$Y|wttUD;>Apbzxbt2}2z=w8!!14B z`-Xo!G3B1JyC1TozHLAvMTd;XJdN}lk)e#{wJSf6e!%!q@>PIQY88RX>qp=Gk^s$s ze6&;DHFTO85Oy8uRG#V577?YZf+|3J`hHX#7OjKG`u$^T3;~ieI_Byt z;!ZksCn!OjaP_aMIv^jefvPR4bsoEC`-QRjC1oFiY??-CA(*O;w}}|owiN|QN`52W zBd_wOTvRLDCxUn=XXpFld|~u$)61Erj;x%+57~gt8{MM=>UJT9yvLYo+h!$V&7yjD zIU}!&1a6wyh!PCrId*r4GrI0Q=X;~zfEfm5W*Ea_u3nZ8p7}fk5c5sonBuRB?C%1n z=_rnCwe%Xsqcajlh39tGu$E2RfB0wQjGqc56qc zwROtQHh3sEW)s^Xl!uu(%421g>_qw2Do)nAeL$`|ZPjspzsq_y+0Xe~U$2qRYXumi zeE$Ar9osUG(;ex4?5&P?cx|IG<<}*lEOy$?-J{)VP|Igc^Zj{WoQk9+8B|xJGCfg9 zXa5=-+#6DK$5GVkxpQ6jkhPWd%I@5`EpxwRy%m$vTJ(^kDBwv+aeJ(H?sy|oQj#Oc zW9zMHzs%7iM<2|2?3X!4KsRz+x8-kv0mZ!5qB=*>kf+w6(_07GfzaAI|9~}R zYE5YWkpL%${92BTd5!(u{9e)1QxKIzq~khlPKWc@eF74o`+ok+I#_VYtb;$R|4PS^a1LwJ4q6O$P`nIcCpj{S7y{prL_{HMDY2TXXLSH_ zlvVCxHbR7pcR~G$v%vV_LM*2g{2_sLO11N#pUh#&eZ%xRg7$9P?}QPf&q9G6a*>m zH&wOofQBKqZoD*whhgXiSZIlIwgJ6R4k2Z8sK5lm_-jlS;VR>~kWDTZ>>~8_yz2mg zedw6!+Qvp=!O>{tG;li8c*9z~d%?q7U80?wE3CfN3g4>nbHYWNUcQ0T+w^W`ySzsZ z{@^)Iaavh{uX&0&)58qe75iIIG6Q*;_h)9B8N}_E#HWchUyP^l{&9> z3dMjyhfrh#pVWv4NuN-P;{mz0ApuSZhfwYm%7W#cLd8X)kQytg^th{SoS?oFWkQ@n zX1q+O?Q0nD2Wy)!{7)Sz<02qEOgt)X0qPNB08#|iZ%m_C?C!s$0qkv;pFMm2qFHJi z{cgB5^`co~yPSE^ELP>3+?VYRv*@B(h{Frd6jC@lUhO!8bU}UfvtWyd3A-55QT;5^ z{qG@N+3}Yf_5XFlVOF5L{}ua1Gwm@`PIKspnQ@rqeY}k))`z8G4Ehb*cNMaQA14v% zf2CQCa0r0zl|yCh=G^cE;bfN6_m%9Ax~RH~3@FFa=Qt^L7HTJF}3B95dM8$55D7uiek3b_|pc z9_{FO7D-)QFwC+IsgX)`t8sSL(4E2#uDrJT7w?h$j)7+fkNx5;!B1}qJ4GF~XHlfK zcJLV5L)8EYzyo>FU*6GKKG@%N2&rfeNw#2HqE0d1s%>>Bs1EP5tt1FNd8<)?W4)uk zd}oPM0@YM`R|S4Mr4#@T4>i15HjgLh9@_Sf|KHIP_ANI7cw%b3H}18Wf+$xbqF+%p zqWW}yb^0i1knaQa^~+%#;x0cgXWweNnfLzDy#rBR_{Ym?M8}lM^CcmoojXDPT_p3Q z9_4gVL*p`d5WBi~zbW0Xi__`!nfa_LAHro`W6tj%Jq=2ij~ntcc?rRJSlgyWTpY|> ziH;dIqJTexkTRI*N6FIj`;#^#!c)eYm^)j^0S(IgF>_A3jWd)?ma_DlN)~}0N|Gb+ z$oBn!MwyzV42A74BBlVr9t0Fc9t~jbi7*T8{V1o=1luR3hvm%aq)y-=;zr2Nn@_Ld zWG+q8M#0ub97fC_@Hayz?KD)=dqvp-Y_EBZyvm$k2aSWCUwH;{GV^?gMhoiEQ2=caKOao) zl26sui$ayoo5}DNmk1~(OjlL)lO69^+d3P3v63Rd^Dr!|n|Sddshd2|f@*Zn{h@BF|&8%p9Ix-yw%B7Hp?yB1Z=akUqt=Z zVQ7}*$5mm`9nF7Zo4ff_k*w=jABN(Q1usvjnckb1ZZPj18-uHi#PX|AX%|tB-4r37 z%lP>m?;HB=9QY2DMl^jeS&O^ETq?+zCo_(Ln*{1?$=m!t+QW|98H|6!@PmnHTE)m} zJst%;)CFmESH#_6CPtot66MLK9D~h+mRxo2YB12@yd8QIV>=9bNxPw6Pz z#r#wqMfQz8G5s1u?#5vH7C6cNygzb}n+Vq-^zT?sTuZu}DJ~Autmj)_E4?2S$&&Jr zo4Y9wDd$ExXc4-bJ3xAL#XaBn+B5f~f7tP8$ja4<$~z=$XEPR`muW&SC~6nh2soib zPh7(4@;Ct}!*6)RL9rZjT)^w6rzz@EIKhuJ-uT=E$5%}3VFWdBZ{p}^IafLWXR^jv z_^}diO_!R12DAE@m=iD>z5>!$YmOE_e=+alv=i&;`9(TL<#3Ch;~21psqO|)09zZ! zX@H=eSL&9`CXvndMD*?IWI7uLBm@3_C{k>;ECwk;9+n11ag?N{J`sIuhMopQ8s706 zf?{ue?#6MIt~@16`4UX$9DVmZq7bJj3Q)|vR6SjhlEC)e!!l$Gshb9=LrfM9Fe;X@ z?&Z4fWxCd7y2fQX{W9Mi$^+wrIW#*bC^L2lrn;H|p2W8(E3QwF3cJqJ#G$?3SAX$! z>a%!Fo5T0sec$WUp?DpC@W7Y-lxCvubwLz4h7QHk*hW!^I{}lu!For%BZ4xSZa|6v z&nm|fYNnAIiks0U(-uHt7r+Zw9qQUf!(=Ko%Wt%smryC`+1|7|)cWH{Z$J0fPoCJ2g$>`Bz2NNGi6ID%kNl%EB|r5a{+S2-PtIa0*w z`jrTp2KLS@LBI^>8$b_?BO4f+a(ZM;1lufQv{Sp+%!&@Fe4lx!%)K$cL=BUb5PTP(A91WfdW&jKet z3tZo`J^zThok4tpmgc`p0uVN=zZfvu^*M|pmG$G?+qLb>eT}{Le+2;XdM$J1vZh_o z6Z-i+`@4@L0biUhO`w^D0&8>eQL4zEH~5JGB4HuRqsZ)G9B&fBm9G z$=NUJ7nM99m#&dO$PAH)@>Ygd$xP|^hpO%zLRU3Mw;fU_88Kj(dYXB%Il8XTqVOff zSmuZ}l3L1+-rA(u`Za1|-)d!Z1T3!0GO#&x-IuI(g48OHuDa;HKg--tQC|tC2}g1y z!>2P1Il_xMA~r`nk^?p?xgjY^WY8)z8aO*Z!=O*jS(e1g<4ZJ?lU8fND@q!BkL|O! zlCipa2lhdFCZMif6Cnuuo zuuu8tx@V>;J6d)4iB9EA_PoWRf8K*$#rLa~9%+Q{En6`6@HqtyF;7XXg}V!flIoTR5vJ2ul>^i*fYp)ctzi4`ay`kQyYx4r0!CQNyJX*!pR*!|~ zH@r&xxvX@z9oPz#Z~%&*t6!_!|E)vnUZp2G!IK?@IBC(V_iUYF@|X88?Q*hOB(lcA zezICkGn;1wytQ?M?Q;JSK1#}=erGIGf1utIwIph=@2MpoOQ^q9?OCE^6rBh6Xdd!N z1@jp|A|@Sx7&F|eyx`q?%*RY$zHb{=XfnTd7UkNLwuclRWWx1}O<8Ia zDp<_M#c0h60=^gNe#~NnUTe@Q9N9X(W@~zF`b;)W6qp)bD2#q5fBLpU__fPzFm#-3 zdRzsjsLTzk#WXW5t9U|erl}>%6PoYbXV&=oWOimQfeGL#4QFR2ne~e*w#&?kO8f90 z$8s?BmUnR(uKZ>O5a3PutiypCk6X|oi*gingGXECH6&)$}^r}Wh6tKTul6Ek_ zvJ<_+QJT_=l3Rw%?7Mn)x!_5WUXuA1vb0_ive;%pwwZb|vokZd8zHKfGwnh&dTGty zh~SaRjY5ZA;qAg61fNyd+ek~U=BE*J$eU@ex{qHHUx4b&WWv4Osx0+@!ZcrH$r4-T zAtQ3eR-zs*(b+yy7nNoft<_!F5rEV07yo8rMi*3pFB^L4!uG%y{~kTLt!)~UWg5~Q zQ=HqDSs%dBoAX+TS*|YGt;|ev1CIhBy)<{JH=I{kF3du_W%8;}=xplA5 zm}z)NZ&S>RAoqMmUx2Ex6%Gi@@g)qvT9579Ao|RWcH;pOaa?$;7Id z@Qy42+30g1aB$5jkWlqi;svt099F8VuqHBP3#&J=b42<%P!Rb!0c0LFQW*D&kT^#x zo)*3M?aLBN^70jHMN zeNZV(U92j27eq(?jBP`U7)>_0OVJ<^M2isd)Q~O6j-d2HW@74MHJUHnIqaYA6WfMQ zPH&gB`RtdfF}TF80sWkXzug2}?QbQWbn~4IwwthCFinvyMVmnv5ogwx02S&S9CP=AXCx=*L$YmyavJOALMx@};MgbGG;{jTd zh|nyG%bL`W1*D$P;h^R}bYMPno9Q8p?RoX^IaPfV%%|@$!#`{rIdgCLy=_M)#5&J> z-*ou8yXHdC#uFa|;Da7e=Xa}_)mV>9BA%9GH44`QY~QUEuJQ-HT-S|2y(c?Zl^>HL zXCr4n+Q!I|TE%|WuXB{^<&L=`$6T?!F91QuKgbG< zy&1i+k8Naxg|sQY9zuqc#fHb{8=3`%#X3f4SfO@Eqc$}=9)GBC(po{`!~!w>XBj%sck-~Z6sFX z-sMMieX$5=r}C^tSws z`JNH^&+E;b`rLvGmn%-}4h&;?kElIv5>f8bzuy6^?)7<6sH8h||5OsQ1idCZ@js|e zw)fu4!HJ^%|6`);>7-`+`3f>evh#{N?|GQ{+NGGKYO(46Z+4i3g87;42#)%*xe=2( zFeSmc!6cEnS%bN0Yrpg6K7!#lY|uO|78y?U_Gp0nv|Y5V`H(97Rl)OOXI!`J&>DG9Amq zOxz}f79tZd6S!oYXV;m^+eLPN1?n7{S$TcSJSz5yUxSi4hqCr_q^ZXgl^4(5$$lL0 z8{2vpbA()grDf!v{R+vq=yx*NC9vy zDR4&!HpL?l*)3#p`hil^_Gif>R+r?9N3((a$YYC3AekI|I{pbWk7>1c@ia_0Wnas` zs|KsY#DLxKw-Jsb;Zva`gkqvEt53WYz;5M{aY7rvS=IJxz75C)_@N~1na>>C1((vVFIl3l3Y0@e`Z+sI*9)hKV~F2G}*UOQyR}@G(~tAfp40`y;~HN z36Q+&p4kv6{J-Z;d@_cm_;I&1k#X-OB8_}u5nze)rQLtiC0w6CjJxk<#y&_7OJ z#HmRvqvHruN@o4iw*kG66X>nFbU&k+VN^O!UzuSf!W`^Xve_|4`|louLg%ir)5j^7 zT%OkOS)lo;3e@b9uiPl(6yC^e0hW+U@y>K}e(y7A!M)eNrxQv_f8#Txjw_watmq5O ztXH8U(%}0)@h!vm9(>cWXv0n|OsGsXl^2Rl)7|;$9c4nz=yJ95=)HPd`9Xa=6V>#6 z?6ttPuLBo;{lbZ_aQ-F>aQu$p=Q=WU5Q#Zc@}A4D=-4Pt+Clj0*nbC@=xY}O7t&rh z@o4}rjDg6d8ooSVHd%VEFZQ3L+^N8Yq!&(n{IhZ^s!6$xzI~?xoD|#*{IsxIdtN7; zjtNGDu&Q1O($85evj34_DOYAw3MZX(IkfpVZ^1&!(@X`6aMGitiB~cjCRBK&Vy8T1 z$IOU^oVTC^fpY^9LrEP{i31W7nIlKCQF^LE%HLYQ;;m6eiJO4P9(h<$CX_pbvdtCM zdm?a`!Xnb#5FuV$Hc+fkp28BT7x`I+pR&qZl*UBqn{`|&T)*7z8=ghNypkYRDy8yE zI#{=W&DJneJZ`!D0^x5wLY}Z(L!{Sl9%g2JjkWm~K4@t|cjBhhBAhMT~C;Eh9 zPn>8 zX6nk$P^(aC5QW37ZpI^YpIg`?=4UP;@)OBg#LrzyOU!VeLl>E_D^cw3|GaiAaN)lz zPYj}Ssqg6W_m9I*X|Rh_X?LMy#&@(Ni2^?}1SZq|`#ESgSL>lgCew6ZugaXMvj2*7 zE*G{wg4_dzB5xMx=qoGIK7BS<=$`&?YI;2i<5*}G&-a5RSniyXf$mGg&-Eqd6J}9%2xX*m z_ZNy*J{z@g@pER8*eaf~a{2_E<$Fo%=LQSJQh6hQ{QqkblA;URc-`ObS6q z1XTX{&oO+XwL*ELq+D^Yoe_K32-k)^jf3`%}A@1^b{JMT7qZ z;?j)|aAs+kQz$JHN=*4T5KLd2k!WTkv?mkHbZBBcEqRqP(@JzhL}xoG^PYfD7?!tq zo$ZEMD9(Q>vm`1hKdZNZo}4##&&%oPQ`#Tbw9^LRuWHmY`HoeRj&bTOVydoFnT4`> zjrFD*{Q@dK%YnbR^!xw~m^ZNM2C;4Pq9unz3lNbdEz@9C=aauG)CDy2n8G3x!4)BDaqUZxLc`L+!NwGrnbyV8gcexRs$96>#R74}lKr$~ zCWRLmZb_#wdj^H)3G-!3IzzH?+}EuDXJBS<)oeRmU!k@akbavgM%+tv6<^N0fEGa*28=H$?ZX!h)=B-*7`tf3#I;a0LG@&xObWRK^;yeux}Nm06H(c5 zQW;v|)O!wOXX@`O(=qN`5qe{`PBgR8*zyiDV@h{>AT0^=m_tyx{RcwzK|Vvl?2FSy)ida(;#S>>CjzOHW?X(uwzruZY0|5Vy!>W?42|hOB@CGSDpVKl7 zWzs6KgN@vx!(kiu+GP^Jdc_x?;PNmZJNfX2Nu^Q;Eysn>_{YV4gh1(YX55-8-6*ji z|5JY$MNPUTwIBOaHqoV!7cUQ6zPNAy6I`Ypr$r9gv;EkK_UHMZVuws_*?1S!hGVxN zpHo-M!=Uc0J8eJsry8tLb)C_E2v6JwPz8=TOw3i$W1u;RPE^!zF94$j9Q>FhPF{J3 zKPXkHsRvSnY%S~}G2p-iKe1~g<*$vEM+^n#2V1uQb~1R?OKeg1eOukpI_`5JhgzG@ z03G1l6)<%Jkkqx-U7N0u(TyuqDe7VK~4skHb&X)!8BeMm-d_6h7W*q zva~`0m{j}oBrjGv1(%p$ZU1m5!{dtR5hlx-rEH&`Efv^5e26qDtc#5O2W)#Os_<0E zEHyrX!2JNu`&F562`Re4h~^4X^PnCtHz9iD@^v}7qP(gm(TN)YX6k$;Pw%(f=z|W< z{6_T31KJbc1ipgOX}*%+^l$s%iri0lz{~}A*3Y}ciM)sgIuXs<^Y09mxjNw+(wQxw zTwXkQiJG{bu32cGM|#dJ96GBQ7l^+vjtHv(X&gbN19gK zQ^+a|eStvlK~n3I;Ru=80fc%Fa%z_XLtp&V2gs58HB@Pr-S`({Hf-f_^)iAu!7?d1 zJND`=21qUjcE5`sqiv6D%eQVTN>@>f)WvN9<7PbOr|U8{lbv@dV8o~(VF%*o*NvIN z)Y@OXi>@%C5DcFWCE-kU?s61v`797L^P&EgFgQEBEo9dHH$!G^To1x@ULf$z$AJKM zAP~5BB4pO*_>ABKI4lO}gqKPRCvMcXeP5bbk~k%H;1N;6A0|EVXq9`_d zK2?0?_WHKpI1;IKdvp#xx!I8;A_VdtzjE92 zEc;h&fmBaxX{j+Bt#yWtw*#54^d56+#th~9Hv%(Gi4AOh zN!$#hWNNJ;ZY9`cB6-N%yI6e|*xXo}a*zg;Nx zB>dEyrxj~SZ(0Ipk8Eb+W6-6fN5m_gla9U}&Wa|~UDQkShy2hl5HY*}gJF+eEKXaAO{5YB{-+xuD0xp6rjOdQ!<;$}<9@V^cE#6aN1z~Fnp8L+*f7tqOq$7m~j z-6!J4APR)K)Gl|RwsxM-_%6_S$fZ2Bs?59N$QOOE+q#l5Fwc(*`uan&rBvGD3{cli zLgt1I>eeBZ|6tW&!2)75vj8`V?fX{%ZgM3xQvTA76|YhaqXCy(`?=J6sB_YQ!Z$^b zR7v~ZlX(3A5rEon- zsKQDm6bbC{M4tJA!lJkeI~e<4?fBIxKvZiYwOjG>$7Jq&%CH6E#3))a80hbX*a35= z%P)^zR>qWXq>C>u&`WjjfTmY&r~9W->sE9a5|0l*u1?xU^!NqumaJRcF>i4}$@^@{ z@d7uqR;br*Jm}TvrN&*IuMyo*t9@ZvpMuiMygAN({gOH{n7@7nH89DighFAudzaYn zpWZ02zeM`x5|nhQ7-=56>5Ve>{Z&$@_5JJ#RAHapsF#HLJ>fLYZ{6p%! zIfH%PCPsPqja7=$9Pgx}a+aj0;O3dnj7JeG0Y8IM+>sN`atCMmTCy5BFau&3*7lh9=D+nLKM(0 zPd|scbK{uFWa&gYVrI;*l*qVW?=vpTu#Um3(dV6$rYggzqiT)9F~MsU2nwWd&RzAnczJhqu5u=EkjBAy=0^#Qf!W<5K9h4vfkP220eG6;fq?(D3|d+BhCC>o2F^5Llr;&NZ`%YE3)T=E=jp3^ocPw_Z6UCrW7K_;wtyhktm`?=zR3PQV;C#ge zBpvi1^XkfnLvKD8t+as8yZN{t+l(rCEq39J?>+AE>3vBmw?v zCkl<0UFV-xXjd!x+(nGio;aKI=iON&@21wg?RdB7%*2|wPz{*~_j_ok`gH;mjnLI<@0Q(h=hHqJOvYdsBsa9jhwa z6&S5hXM+__;bITHxcT{HB9KJ>E$r5H>kfd5UdL0|$?y6WE3J=5YEO!f3wuP3YCAUz z?Yh*GFv_5jG4dT@S?iB{pV|GrQA(ua#BQNLBeioCs~>3+D;F0ONLR<%Sa%^^@UEc1 zO07<}rMuZMkr%WIyGW!TqahhNiLOCW^I=zkH3D?hZGf26DBOYxG(v`fU=-DEX>Ffk*M8jyml8(^PmRF=Fo?0K(csiuMv7W%$Utw);$&fgA(>u7QnZ{v&uuFAx zazR7hvk(lHaC5nB+FWpDj*^I9d&3EI{h6@W+DMU46Et2}k$)cI8V}mtU)RNn#$<`2 zemHCL%IHU@?W)y%<^dt1v7>~%WvkY05y~{ z7bR~_7tP}wd5ry&8JTPDfX&0Q<%*h&#?_t4snh+bGc!{k_0M?p*Gv>l@i@iGjbctt z=8#J+f_NGD@<;LVEB9UdVYB(B$>`5rm6u+d4bT^`!vD zigC%7Y)bIS#NON=WsIW)0kQ13jL|^m6P>44Zek!~r0q6-!sJv#%lbZ<#3!G4XqJeX z3U0tBlRD(#%&h8e0W-gcmsFiRA9M9{^pU>E%SCa>ruxed7Cgv$T7q4C^~Wxboi{As z$Fw!i>CA4UaI`?3n$=+y#oT-(q=#-79pmpT`t_5)oo@Z)+~nXeNUFM&;24qOT5roh0LK`N0u^MKsmFN&MYQn zabHeI>f#QjfPEa-bFevK(4>qR2zU@xL~%T1)=9|dZ5Y#k24%e-H$T!-7eE*&k~5i! z{bn>|(iNG4zBjrew(mhAXI{}<`E3!b3bIt|r$0ApQ2)DtA(x@R#oZypW%YVkZKaTf zJ$0c}nv!0W-f>Z=Iq|Q+J(`&X=8;#RtNOM~!NJco6Q)EyQkH-B(%gSy7eNX0h8m069k>u%aiaU@@Fpf}X^H

TzCfz8=?V1ob(AUNka%ctcd%Ut!=nqM*(ai-_uOrN6PBl~e2-2KYRx zaZ0O_wOk*wrPcOHA)=&SihIo{HJL<~ljy0^nPV`Aw$rBRI(shvJ0AA{%qUb6 zOpP+sA5kVMy`<{$__?dIl{Y{C!>}Im89xw*OA&kTWmCEUrYS5WB%C9*@8#4+T?&W1 zMnc88WU=q91J;419?F0NuFujTc`i9Da%HhG-!Zpsiy>4}>#|h!i&dteF+ni5s0hyR z6F|iv56`9&^5&|sH`|+-@2JIUs)FR^SEx)CBzMzTZcrH0+am*j+#{E^ z$KEs^;2( zp2i=S3}iOLk^ChU1F-E~3k5`G|lcr4KW^IhRH+D3DV* z6Y2gnaHkb?=)y}3VpjeD4ggjjp)*aNtDa%ip12tJO4R43eY;^*MN9)%jPy!=M-{4z z65M8X=QNS_iAi1&^LT!WO6${JoUF`$N;P_Q`p|>g#|0-&LbQYzgSzDRd%=a0x<#Yq znNn+Lk2GkGVEp?uv1C4_f1(g3?#WN+_v`5#E33fQGtvG|fIX$Et*(jn zgi9)RPGPr;njE-M0j!1nxks>|kn@P_<>U!o*;LeLqoW5dJ|Wri2-vrpv7RVpAi-yXY+XD9X0N&FtU&?0CY`Dk}rUbF^gh(GglT8lC@y|r( z52=sVM)i82_#ZqvJmmNL50IAKHQF6mTzM7sGQ_Iadip&)xOKza9j6**sEB3G`~kecjsNM6|!63IKH z`poCPW5c6Kx1ynqsUqzX;{M9V%&b`t4$eLS+xP8|8Lh)-7WhFUK2PEKw?X^%wL;^Z zebaYb6j&OCRh4|X#T{{~W|>$^UNUMZ-b zU$LKu8OpeUJw>O^78}f+`h$9UiS-e^Hu{oF6J2GN3rxCjPv1TF!&QJNkGsyjnE3JR zc~vka9d!13Dl4wQZTqM^d0zUK-H-<|*Q*V5zfz#rx->w7y43LG;tadx)6BwRu9*2d zOm!;(h=Tk=6=&D!!fV#QnNh+{_8n8r+smwhF?j5+F#)jBjdE|Ta`o1%Zx-J0q{N(? z30v_)?LrioObWPq@8(Is2$kP2x%98*?LQ1OoZieRoy8!$MniDN_WjC_UA>2Dk8zov z(*my9k(b34CzbwY%Tq8FV4_Wf+dOZjwzk0QdnnSTL|w=_^?2Yl55XCO#N1|=T#G#f{VqxL}txtSMTM0+ND3R z@(I`F%8PPEavCcSJy9M=j%DQ`%x{y`telxTFIfmz;;m zF)7*HyH=r9Ogv65u?=2p*|O_?2`&~cpe+BQ(w_X*@Bw^R?#;(56 z(WTbY?ry&$R^;w>v#hbtO%21OVju!?w%B|3^>p2JqxPaU6f#n+@|lkr1=4|E%I*V< zB)G>j_PhH=TDwhQ#L$#pVh(qyh%&CXY>*qZdYVr=PGrlB1B$ElPu$?v8zzZiqP#?G z={((JIPvfMU(r9VYsT8v}lZ(HiXgV7F}*W-GB>P#r*ZIm((X0 z&`>zTRu$W0*~K73TRdXf{DI`V04c;vt6K6#h}p=XR#`OtxlW$(8B%T$;=afXa2juy!dza(m$ZR`7%)7 zg*k514ebV*JN3p)gGVTwDuY?5cC*BLVQUsy`j5|YZ-a0v5W-*XKuJm@EJ_eOfNF)7 z4)pQaGtI{Ww`h=4Fr8*C2<{`DMRnRe{b3d#* zc0!`2xVPtJ5h0%m@}^s0YrPl9te-r4ZZr2b(jfvQwb^ig?_-L+C;GAW)QLSHBBlme zjFvp&z|-)}+FSB-53#s(xa>hlH@B1Nl)Q+tRN;f|bm2rtaoF`@4O#T1cZqequb3x( zp1kXeh2hT-Gc@mQPoK#C7b2U$&Cl0kHI~&s_$De{y_HP;o2f7Wwd84|x(`jJC5x#x zN%0432NF*|5_N>p9(|N{uhyL{nGr>AS%^DmLw0EC?B7uP7OyQHT-&MiXldV+uWJW= z?+qV1Qg-?`$7OjwcZ&`E87WrVA1Bu`p*@ruiB-3HJr>&AQ+cHuthFtb2}?rxno7S5 zB|^-`LNXhr*DU>^7>=Z9mC6c{j98H7SzGDTm8iTIHVUne6GASb=WI#JvcgHeFie^6 zy<7Tg9UV8Ax~{i9y?S`OvBh%xWYS~rg{dLO1#iAh=Cl8-b~CxpPueaR>D7^xWgjzj zzFxhjS7(2e%x9A#R=gIbx^35kvVK?r5JwgQEyWN^!{+l*gIp2G`l`f5UYZ2 zRBDeSosQkK)=Ssr*>&UxVt3Kb0(yb`X?Y{tXaLdNJ^}M+pG-0_iLrlJTY924YZGlQ zz+F?T42(2r)Y=CA7yDl82{F)-%=Vxo`2li3X5J2%ICrkhEyClq-1uw(mHGWObNoch zZjsm5=Z^QeDf<&MQGIdvw~T0^`df2>B3I%SqX_f~_G9Z)=eTWIZVpZ~^);SkQS8|= zuUMDv)`x+X3e!uV#Dtt{q3GLoI#-+;=QhVr#e13E>rJfNtdVKUMVWW6TN5ZxJY`Ls z2%Q+jhuT6|w*(*=j6GV0YWdoTvu7$VMTrRqcp^eVT%KG~0KgFJDp0do5^5isGQ>jZfP2MDT zL>{fYSm$CVQ##K7?^40IPkdU}^;r2TM;s9j@tQ|_%rc|Ur(bV~n?;y-2nU1u@Q&;2 zf4^K0;D$nyA%_tgM=e~s9KZ5gh(0ZL!ZI_ok)k?lR(+Em_rgX;TC?ix(cfV$37}6n zV->~TOcYz`Q0-0rOvhbsnxiQiEF7q) zX`|4SWX9d3ICwqQ5}$Qtw1Syp7m`NGvg#{8w||(!NOx#siq_pIk;uw4N5M$pUq`ii z9p_)ixrsr=i)jw)>N4vydZCA1j*8$(y_57%afZmSs3W8wE!R#nlX~dr$TE!KOL9-= z8`?-HK8?bTgX*Gge^Iv^6Lg$LuiC}@98HV?{SRAKtWoBW^~fBjNa+v?cSJ-&kbgS> zDckiuubIZ};8)!A6CTj!yjYi^<1U`N>=4rUZYm{BPd`()&4AzO7xgE%2q^tYiGVVm zOeKMu&^_>(u?MULv_`vl0)u-MGz;fmFBN+d$w-QyST-=S@ewYuG}~Rh4Y3J?H9o7A z8Nr^qOwLNj2*QAuuJ2yCR?+mNLw53<%t43Nb#hWJqOS0;03|x%cJ5?1c*ANln1~(e zw8KpmE%Uc>%;c5nv=d&*QUtK|&Zsx~>uS+p-Arau8SECto&?f<(@t1F)xAn4vp~>8 z$c5YyGNPn2@DeR_mP~}zm6E?;yydIF0!2R&GnFLBJQ~P35n^hr^vzMcv*P;bY2W3( ztdz=Df!n25n3P zs_`Ohmxmbe*z! zy*&aLn!w}>U23@<&Nh6@WD7MZ{IY}Ao>SG^Q}N48=h#GoT!n+ zuBsw526XfwEAV|I0C>muw?s0-(GY?I!PN+fw&Ay^XE{D=@F^dw^VUuC6i9tkoFI`6 zYq%3gG=wzV43uc34c`$$l_H(j%WXl`2t3?L^a|1!@}7O?gAHGyCt(d&`1daJ?|t@* z_sDo*=EJ{Ngf>g@S%=TYpW07IWYvj=iHRCIu1lX2YSB}@pNMq2*a zI;1p*RymqU_gqAdZ$re$Mx9+zg@iFl9)hWn#%1F%D}w8n(o9u3{+m=f_TD~#ll`va z=kM{I{_#D2tV~3MFQ6l{q;A-U(JC9-Fj_JG_6JC1|K}Vq3|+a8Sx<->+-?FOJQgVi%bAra>y+HT=^;u$EI+u*=~fb7RgSGfIl z;Wg+tF$oU+R8ox4vG)5{`7}*1ZEQVaPky5}AL%o&8Amo4*u;`W2yC?d%HJ1$d)3ev z!HL@uh$Dj?A#JgmnB$@jAH-LFBEb>^{e#`(&1d+!>;x?&^o%m1!%VPXS#BaOPG*$X^+VM!ZI}FKI#YY zc60^}%ZZMc{MgU=ZGI+d&ny9>%*{M50gcXtbaeXd8e+{{7|VpOe3&kJywB8naW74R zJ7c)O9dOBSCTnpsz^PfB&*qOf{5i$%0eeP?(Z|U=dXXhh08|PcnbX)stR$+6JY>2SV-;!18{q`EPrbt)5a6k%6nU7SQp#VU*8 zp}}q#bS7nx!mY_n42jPY(#6%jl4|fXlV=#nal(Yu>;3jje!RQ<4!2+rf8=!Z`E5h~ z9tCfn(sC@of3Fw{eqS01ekb_H<3I)b*U~pu4gyc!tYs*eZ>Nd_E1sq?cmsVZSCWzG z%`(uoGWsJhf|fzbOxugH)HA2Ud^I@U!TVziwZaqVJyYZo@&5oSB&Wk*OQL z=x)8ogrAx%U_=Awn2^SWpBLX=-<4&bK*n(kccQ!B@?S1yqW8>(DLvJZ<@aa#bAB!U zY`bjFVR3z{pYYELb3%DudTTnD>7+}QxkI0?%Yx4cBQ(Wr1G^0BoTX1%ocCFAsMc`b z`D&lz)px4*p5-K7<4|p>r1Z6+f0NS_>to5gD?!M4JWHtJoU5MZHKni}vL?>z5rXSM zVnz6v_w(rZd>KXrW?^aDa*r_HCNz8(v~|;hD&Y5C+OpKxxkY9n z$O!e3lbPLLV}i^g3L>y8H!|}vL2WM$>9>G4Gt=ScZkXwb=XRTE?YSq+^n@mnnVmay zASe+0!`bXmYABbHNy16lFw67+-msj!k5?dB#XFX8{GFgZaL-JuEKl6SyOtYwgS}bu zF=AfTX8OO$6$OV0L8OC@G`j^R#IJ*%$|lmfCS-PgbBFqfMJTz@@7X-ni<;(Ym1__wJp-W(2$-{}!>R{R2T~1aPmE~dM=I#Pm z*pdmh2qKrxPu*Gmls9cze%H55#33F#!OYMDw5T4>mVIc&U{sZU|0URUHE)H(H1`{+YB){-LB_L*exc*e{p zx#)(gfsr_^BW}`gE6BAyvj~`i?@8^(uR~7JjxNA?>W{u-fLe#JREuz2!jqQMuwxI^ zZbUSPmo3}h}TsTG1H;xb2{I#dE>~T3Fq#<9j%E@VHKmW)$}^%*a@VD#m>Uu zSav0V7b9s2Bkul}fH0cNKKE@j>SDZew8PfjRn4EO+jO}4ZeJZ0^$J<5v+(00LQCRtp**t%6 zI_+=8`Z)01aoLZ8m5b`nptSpCv6^q-^%FrZQ!GN?V+dzd1Y z!4%OTO>+IpD5%(gE?a&$mA123pN>AIXTxv~1r1r@mg^!Ewm3!h$BHmZ?b5q1 zlPT&A5Z;l|GO~50RYmE#>TUXJX9A1SpA;BGC#!kDY5Cw34>$>eGuoF42F05zz6+kk z#j}6Uh|JkxX32zwAAXH#Mu0q=NHAxPw9g~!QTGAEOHQ8+JSKvKgQB!$BA+%Ciqj6; zNhZAQa^=5Ft5vqD%mu=gM%GCS5D$9iYY3EAoe_?)+gO{K79*fCNKKPzA5_}01HWf! z{gB3;3JYmo3z%>rpcgZWAHib9b=Wfy*xvUw;g!iMEKqbjr9-zb4FzH~zqit%=UxL6 zZ{ARBh|3_$83rs!=I(E&veVIfc-UF#Fo&6lv=b`Ew#}BnxmKev$Z<81lX}G;;e;!L zl|6a$IRGI!Ttdm90nlj~#U*+|PF~s(fL)3?oSWmo#Tm;)b=g*Wg5@Sv{Mt2asZxp6 z8UZ5?>tu&!LJusw90e0N3=5`SyfFIX)1>lKr$}=%Pv2dQy+jgbpT;5B6b|HB_jeC) z=Vo}Ckn~nkJmKhRR#-n|#^a@L?vua!d2~VcdvNsPYwsDH%aKP*Txr ze~%sbox9a-9C6pWVSSMbmmS*;CcJ=Fnz#*1j7o6Eusx`+zX7V?ud@2buUl^1Um(MY zqRfJX*3@#{k{2N8C_neQDc6sryjTA}*4{m?sw({--{*cfY?Ygbs|}!so7w={h-M9- zA!sf0(>-MbYc@}_8H7!^GR>#PmfH8;foS!W5>zU zim>Tc#;P)x`evl77H<24S>+q{M)K3X8M!z;#&72pce9F}Ddm6RP*UnK_I%vg zbKmFhK8BsC1RX~51mm|K1HOzc6giR{`z!Xs#321&MPt18)puL?<;r0 zscYgIiKbNjR>rezc~_HU)jy^2EbXh3+)2I}&#S5L`IQ7RNy=rX%4Q`hFwA==%DppY ziM^5r<(@Cx3IBDC7CD4(_dMeI8!6QwF!_E0L;K|37f*HKywjhZDKboAs;|T$J;y%2 z=Qomr^RAT%D_C*s#o<_ujSqXxNm9OM% zYM;lt#CY1hU6pRb89sUAGV7GNfuT7o11ZrSx1_MFKE3c-cDgEk zU)f@4nJKWS!32qMfM6Pi#T`0a&`El}tbbPGiZEsTQ!?_7A?N-|r6^40{j5jg?LmBF zvxe8!$&-1XITwSuKhwvukhDpFBw7~3%T(8{k~oIrr7vv;&lkttvXGKUhS%U=9LkjK zKKe;&v*y=jy!CzWUAD5|l2E21V1>=g8hcqkU0vz8S#5ieQ>@^%yJPaB<+yQnb#1)k zeHF69ArQ`VK6jr`c+PZ;bsxh;83Q6&wh>Z(Im|yvUmRl-vqT$MsLqeRQZ~q^*2&>L zhQ4Ve6<9f^Hv%7*L!?fIt~Z=!BFSqc{}=UP-tZnD4TlJiG7Imx7eqR6ObWrsohvbs zX)-3_))n}h%2>p5!PgoBuR%>4<|tz=zKvN7=H-Zgy%fL05nklrePc9Et*iUxCRz7W zzI&~FB`!JDS}NnWz>$QbT^rhQRBMJZuQy0Y#!0Rp+}U*9R{MaCM2!~m{FW(>RDV_I z?9``Ji6WeWX?uWa)UdYH1N;HIJ*1cL)A2rvM0NgX4 zPN;neYS2$RmX1wQk_D17DS*96F9FahXN1a^;-@%ZA*pn{>mat6`r~9>iRa+J@a@^C zkS;nnz=#6PCTY$2)m55TUjg4HDJJVG$FIYiBXSLvTH~%~M!>!e=L6H9g&iPo?O1A_ z`)_Omm`VscmsjE)$Na+SXDfA&BCuDN6EliCaS-uV(#q60dQpc_-=v zXOs}yaGj(Zl0G(e0?KDtaTlwv6Jm@8{?7I})pVBhi93zVR4_qzd`>oY++l*C`G+dC zD*JB63|cH0pJuwYQp9t2tNnePab*V}4vbZay}W8MGkq-9Rvhh?SEiuBrzb2}a6|8> zj2bJ4*=a)Jv-!e}T#^?jspDtvoA%M@ga>lijY$et*E;?O=nUXXyPDnBf;}QFSPVzq zn=ov&;g+BTe-W@w4#1YfW}@T>K2F4@I=s^1%{Bc1YP$-WX zX{(k*CZf5@rhxdC8<{Wfds~;?_IYK;q}qcmZ7r2YCvg`DNCXidybM6jKvPaj3EWQk zADS_#1DMVc&_y*L(fZ&IG5*$LlOPJ(+F0jxX(9K8_-P>@@wTq=!5>VuEjVFdoa)xN zACT@}^zm(-Q(YPNL(+KM3#^;=ml~=j&;!qk9s9U8g2+{S|06q!c40PUeU2+h2G>{d zIW5k>7gti8uHm!GnVB>SB8$VJC-ofm{AX%vIFRA|3p9Q z#6Wy?5%dl@rnmYTxZPbeW^(qJN$F!+p7u=-fX#j8)8UI|gQ;(2anr%uQ_^@N~yX4J)rTcONL|cW-+c4le7f*ctz=T3E;ym9J8{K}NJ1)uA!`p3s2`|kHWQgPGvcAo}P@O)B`?yZ?bAOxcRc`^4;h0*0`Km{2#Q!(SRHP$D3r5DuYX;){gaSJgoS>q;$ z+O026ej73}Sa=HK5%VC$JSY)X>8P)^%i%|8=ZxS=hXe0v(=a-B6U~?8a5zBSfoXJf z*Z&vkht+A`*1+c}>^h-JC9fmfMq^pm+qyap)Y+H@_9d%hHC(z`CrPn;liSM3P3 z!w$mw(xejXJU8js5Nno-TR~;EEi9XntNd~Rcca=Da@kUpzV>{&^2iN&3GdFh^$j{o zI*h23AK%yYUBohXZw7p7-0ggNL=#(DQujoOPv>& zzH!%S*kJvI!8%PE_Aj_FB(rA6+nThmu&*kMS7g$hc)zVV!kn%UxyOpfD@;4-Tc$7u zIjJ3)-^Y_TJ_&MP!(g4X*R1i%DXjZ5|n>a@fMz zbn!I|GYM-6-@n5&5eZSfl;5kbLeEKiLUB%cO%*99%4QgB;tc}yt?*SEXJZ(+MWN3> zssW&%$9N#5^sn3x5i6X}gk3cP7}eepO>f*{*9+e6;()(w+WxIcKF6 z9LAb*`lS_0vz8AkunpNmaWhz*=RE}Xzislub#IY^s$>8*{v6y=z#Xwlzh#VhynLyj zC4rmqYI{2q4V+}eBj zdD&v8*@tQ-LZK`vq3yCef4odT$8C;QSfd$B%FeNJvPo4=ZdT8)KDyA@=Wbt^aaLms z%$R%I7M*eHmd$_X&P>}hJNeIej*zyU$3nsyE!P*RDr1@RvGVG_JUqXqS$!$*XqJ7U z@dvkUVSCoaysOKd!O%Q%MpxTMJ7a62F_Sz=$#J_-v(6b;Z687`5%u=ylB@1IFWdB; zJBN_Z40VsYFs0-Ru`d24`a39;?=9u!xq$^i-fv4awlX-4VOGkW8g2&R1Zhe1yGf19 zo@srB^V8!b==@9S>8>8*&XG!9 z!zY_vpKLy-yGFLeRokj71$d~W-qGYW?JZdy)P-!Z>Z@*yO-$m8jP*t60?e;9^t!u3 zI*co$iz3_Lh?F;2qKgb^z3A>-KTUABPmsW>esEJ#svGO4GGWi=@>|J$viQ|shG&+9 z<6gt+9lm9ku8qMu8%wF4E0XbVlOLu4UB(^d?wsB8n&Qg5J1#Y;r@bXli(s0f^xj(i z8BG`@jki!HcqXw@nRt|Gh|08;bBbm9(`MJ7N6KW}Ol3;D=x+C(wp&)`y~?}k;gav& zD9_1j;V66 zhnr4^H!FL-cdJi(lk1JB;cCB;a%JxMc}74yOxb) z*7FJhI|HHdwVZG|=iqG#y@CGiVzYE1Q+TN3{y>Rq=5)nz&xmMJH@nhBG-)Uyt8MbL zEYW)_`BY)nW{d1%WP9Sa!E+k!F~SuU63jg*CYrk0^@vC)8)C?N)hmUo7SyI|YG41|NjFzXz9}R4uvR`v1`sg&==fjTDZ^J&4 zIpvyJ-?d{%*B;#z=nT1p(WQF~b1}=dux@nOfkvJ7Vo1ir?vfu+kG5d_B^^J#$;;_) zSy&xPzO$XddYw`P8h-fy$fa!K|I8%Mo<*7D*|O?Jw_z-cZ5pq?q-^(c#(os0d|Rg; zcH4pvovHNFUJM@AK@-TUeY{PvKd@aS z-Z=|qw_MPV=}>pP=S=1PMPFz}L1TG2Q|*b|F^W-J#WAz^PMy5!8+RIA%SrP|hM%l+ z|0Y-KCgH%Q#>(%w{l$RIMQcLe_yDDIMuI~R(Oco7j4ze>E+3~~tAknnQ8?E?u~7Af zo}tr^+e0DjV!CL}iPUvsSj{=;m+0Z6EsMbApVRlk2yxIcN6RsYMi}08FAdcHLp8rD z{UMwJ+YE<}&aX*|wI@^wZd-!W&=k8fB5I_D{kdA2*=EwVI1CJ*&hd$?P{^ z9%-iV+X~bqrr(E_1fOZf?6CxM+`||E`9-BW5G~Rb>>ZTxpxw(3VeHuz~v9me5aI_V1|Qg=>h{mKmL6^)G7lxC9P!g(vw;N-ibF zxEjJO60=U)dQM9<}W-#KP(1S&OTlPh+U5P~|~!G%x(%KMY)I^FT3_=r#^8|1*1 zy18ts%ZJ(q>K$ASExPs^_ol(pGK9NBNvwhll|xfhO#&B7w(NX@WvW?_nHMvBL~i8t z0Lz4o616k6S;MKQoKYd+W+k=zR4)EryDK>>p|%Q|G>Dg&+(ZfacAYb|R3FCOqMX=K zCp1o2D{()HA+MZ}4@psq>yNqvlP=)<}O^8Wljk@p2eLZ3UWe%>eN-wpvs+d*io>X_2&LX@zi6Sb1-7J zQYc-0QQx$9DC(5jXX$TPZN@ewmYatPty<{v412M`)^zReP2xhyLExCbcGvP_Eg`(B z<~J5yLtSyPJM4GopafB;tU52|#r>TC)4p`uygO<#+BMG5t^;ptF|6aFtvcL=tE%3u z+3wYL!M!A8&=#CF;I`?$x9e;n0M7lfJ+AAja|SM5ZucrXm&CTKj_lAiO~mzDXWYz^ zup72uI7m-o-_{Q9@VA9^On+MoNl=nk*|iu&l4_dOyL}Qc)7Zbnzuy+D9z=&v`wl^o zUomyixfk&x`@phMmT|^=K8moH!1W&SoPT+yf(??qP(mRrJkwv2AI6*fONIy!`8vab zVIXdC@|N=%7nR=oX5RW}`kds0J{h6PWK$;O%%+hWhKsxCj%&Nnau`=awQSn~eX#QK z#fuKRvCA1SXsz$NeMmbwJSW6EzQW%mjO11BZ*ur!PN6XxK9N=w7s?yocSGFIt)!aG zEOpqDaP@LhMzG2m)1~XoEOuoU3z@}@l}~bWiCOG1Y#_$q=LmXl^|ngG7`fz@v0`a4 z?pw$9jfbR{qgS`ZMWu* z8(vu%6NIei7Q5ya3n|Fj%EeLE*4!W{3{#TdQWE_ULY+aV>(qvREW=q1brRq&lo#L0 zJ`zx;UaCnCjxPR@4TIsJ z$~aoyg!zDiocIwX4<{frHJ3y=WAV|0Tf7}8-T%mUEI}gQXykilBwrtj6!Z0`H!)uw zy?OFwDc>ITNWLs5enh@<0-}6FJoy^2dlU0@(S<)ju&=i4h<|Gf8ULZ-^;$Xi9T_G! z3~ziUz4!J;*PV^$bQT#+9EKBn#D%FXi=AQh!WH$(@~u0b(DGS)&~THhZzCHsb3Pkl znjOv!Q@z)uWILFQ11j#Mi0|UYXWDx&Y;;}PNZ8Inx?pGrKAHLoR*zox6jq%R^lMl5 z#+lnNPY{wTganhqF0MsHaC)QJk=gt2jjl5u3|LViZ>>IkYJLopQLW@Y6yg2}IAibe zjjod(IN&q7_OJ>NkK>_2lQUepQJ^db_47H2#xkZQ>YwVXQy5OU(~c06+tR62rS?!x z5OMAY&Ya#4H@XgaaN^95NzxSGS@g*QO#DtFl}D4q;v4)3kxlM6?maOwW$#5xv@NAb zix!oCvvL})6-oni2B6g= z&|7)H1KON6fvenDq_&js+)6FaRoFI?nj^ZHmpv(MC+K(md1Ir4$85BVXm~`YKN~7+ z*og6Uvq;mzih_;Z@WE|g8p~=ro-P!o$HqSuUXiy^$lJKpVoQSixFl{Zf$Db#@(V30 z&a}~N#7${t<5=z$isxS3__%0oUg*(yCIO7i<_Xxu0J=5k#jD-MJul_k=$g5$%HnO@ z3Ozx~SWBMy@Pfp{nYfV1mAx?x>j>_T1h+}2pBLb)0J)1O2vrTQj#Qgaq?oMXpK?~< zME@y!1NS(6vo+|?`0#nctc?~wE@R^|xn&VBXJc&QW{Fv zw+5gGVN))`q?W#>HoX>ZNo7gc`WuSt`drLEQT*=btXObgyFN&f=utf%y3G-q`BOko zUer=~#$WpDREEF1%Os&ZTc}DFqXUTd4kZ`Rlbfe>J zxRl7agZfh195?OTekG=Ab2D6@luXm0MKU?k_o_Bpd|jiFyl|vjSN(ap)G@R zwKQx^nXl;9hIg6pO!njo$D|mT(s#Dt(#E#ll+p|x`n?;pfoQu)aCGLTNr(ML;4LrlW|GmNW zpA8ae=(jzuOK6}6UN*~yyAJJ&vI%^UMajy^UI6#=Os|+LENjSiO}OgEeJJAj=LWN& z>;2CLSH}ivq@n)LNqDT0DzoFarNPnM-w4(mscLB5VDat!3~9Vwr#uNj{${`{)cx*s zIUiN1+_ux2h3nVCtK)m#Mc7+^>?WB*q?~(M z%;q&@Gqu;Y!DYU$zW*rIQ?^u3Z;t1(MA#z0KGeH?gKImrvBYwQm&H1|;a)1|Qwvqw z<_uqaGee$u1oJaE_j#hw9IY?D#ORaD?U-A9%q#Tx@CW(;`Wg)~tey>T$}r4h%qL^S766EwQ+u&M`DkvK1e}CA6hG&+_*>Sn5_S9`TO=H;y zR)4vrPHilc4XW` zqI~e`*KSXdA;tn8T;`aX6SR+=4LJm4E2EHpyR5BOY#;OorAS;DhAJ8nMTqHLqZ+5cKMy@jB)K?`h^z2i5yH22p{&yx~9NO935kXrXC z8`ma%V-{y+vl>K=0@cuIoio zk3$EVCm~)cx5_6tRok{&vH~y3)lV`u#ooCM$_)qXc= z+im{(mGXvLMfG+Qo&@_}BJ3A{ZR_nTa@`z({no92P1Vp}RMB5_U|;c}dV4nXBZ}`F z`?UUp!!5_Nz3tu|<>i?zEtxI$aO_>L?ytC9wCgYBZ{Z8#5z}Soszux%1NX7suZmpf zM{swz3&%!$Q3O|S4NW@OrTxuOUt8W?>)Y^Ek)yi4_HXdipVh3f{bAO6>Q*kc{dEj#yYdEY1zpvTa_)l=ug4! z=p8UAe*!UC7(=)`f@jOHwXSTN#I21*$f=*ZhZ87%GUn@VxFuo*{t*@URqwVUm(5dw z>XXEm0^=b5*{*~RR9Pb%9KUNv#$3jh8WkMD%@q@7BH{VoKNY#2^CZOjFiPT~Ldegz z?XoWfClL=Vmw0GR5qM~g$QMZ>h6TWIxpz~MYtu;Ejqp&e$V0gz>|_!4F~I)Nn_uL5 zVFdPDw{1M}+p{9v9)v%E_pY0JnA(KTL#u36a=ql>D$ffH_hRpiB3HsahC}uG?BOocXR|Qe z`|BdtL?U5?-$w6I3&d|r1Ho?)Yul8mN5F0v+Lh)kHFuS$2N9#2#a)i!w=(BKardK= zK0%@2I|3niuAAP#Y-7P}ukV0}bRn2+RV!xqhrw<4M{Y-?afa;TcmnD+5DSiD!;I5q zz;cqXupP|`;=s>i5a6?%ddnV;8R5Zm+=o=0h7*P6N^S3*Lf2rSG#sgq6Az{^SF^dA z0;ogr++G4~)^!dRx>^f`*243ZV>lAIGMDM02luQkio_CHOL;i&<+0-aJPZLc zuF{i7wv-1Q4^bN|;?ku&=6mvZtI(BIC}bgzhq*cbTONPDm&fn!&!cBFPER~}h+~K( z7eO8srG=*)zs1g@GIlJUjpEZBOa2sF#dmA0-ej8KGe5b7=Nxq}lz+0eexI>rcX)NP z%Hde-aAeN>@uOxaOE0dk&n(D=*RdB&rp$tuRk(B-6Gp4!v&hz~#|}QTyS~xf+7(xB zceL0YyWVPav>mAJJ^(iY06FY9*wWe+ee9r#chvKa#`o2j$08kLa3HgOn|f!HA~dp= z7wZJ}NzW8kJVQGuUY+;*LadbVYEf4#6QeZj7*C9beGp@G7Yvaq9jmx}v0BHlkFt-j zUF2VP6}O6_7Cuo};kAyCsZ5n*OU$d9TJwT)+bV@X`lvT4BO8LTc0Y{hf)o`{3$pgzV&I z%cf-mu0wg|XE6_y+kIF6AS>UQ(tUq?XX_d^jOp+_iEW}Mo{N<5{u1eZC~;R5@^N~_ zi>?~*S|b&L4ZU9LlkBl&6(Pc2Eyvw5)hXTgN1nvZVn2m#P}Eoj>V8QYi~H>E5ruj1 z8BR}?3*fL<+0lr29RA{cDXLy*ZJH+KMUTvGu%x(~%cC3$%D2$hAazZd_qC*DQz zrTM=X>I#Z4A|o4i%A&+TS5Z^Il7OMcOK?&R{`5tt`bD_%7vaBp!WOX`=~7|nG+VZ_ zPu^x?D(nS9>RKvMEb%kFtYpH=RN-%_Ot8JR7HC$=)rY7i>7Ikiv)vv61~Il%Rk5w0 zT)LKy`jS(7BwyaP4^v#@S3~tuya=altaJZ7{Qj#5Cu}0RY6yf^G;AmkHWt)NJ)N}t zc%AF(XCMpWQV_6E$>Eb(`YZ+%#d|$L3A!5PaWKgeqlpkIVMT#wpb)#duK4N*0+?T| zpqeJEY9VfB#j*ma^cFQmX@m{=5C@seV%{1ts(9D~AqLgOxa3-oe?iO$flYflGKre*7e*1 z!r@s%aN8)5cL=y4NaxZ;^ap_68|c5tl~d$wisTaMu)5*H^}YMnyFOg+IseS8;$~9( z)m-lH{}&zBm@z`_dQbNs;$9(aSik+6ekI%Kzy4oIZ9~HDdnrX&!++zH0d-;|r62s_ zCXiZR?k|K(Qk$W;W0hyMk}<%&xQMd#QeNBEv*hdBnl5@AMEzzl^;Z=2%}U|*_4Rj> z>;I7=84FZ!>j4JWFt9GvC%HW^#Un5#u}-9C;rjW44M)#*UrPc6l@Z)V`nX&7&6@R{ zYt}ccS>Nh7!WD=K*3+{R9Q_}vHL(q{pywujlOV(=OmlWOyu4mmwVn=M*#}PHw6(*R zJ4}h9LVP-N*Sm7p3%To`d5Qa!&NyK-P?pXaMKrne6w7M5qA+b&R=lu&PmGE?MQ~K2 zSVo`g=AIE@rKjivy^VCxQgD>{%rR~`eO#s6`+WGjW}Js+P;is3EUK%P)u82`qA=wB z3+Wg4!K!i1xPHdnIypP~oce^OS;0L{U$AS^d3>H{G260^=iv}nL&ql~lSPI2cFtPw znzde-wZ73Y%vp&(oF;9fGCI|gk{6V{L?tg&6w#&vZBs;I-6*xeg4<*5Uv}gui17AHIkKt`mv;xN-UgD<^W%@(dpJ^z-Y8~lY2y)Kc zs95z9+XJCyiy9JnJ!GTYopmO&!`!}`PHX1e4vj4NhWt+OV@n3ztSXIc*im1_w{7wn zTPn{wasnQSIvg7HZ#}Dw`iFn#^>wc6>lz*H!u56BR4NraVFvG@Lkz+ZoSYmGKbstd zc*gbMao3{``$zpdf%T61$EeP4*15h}*C>2f0x@@}l+J>5)|lY#(OBby%i^e}U$)>*9dXr#v}dYtDv*ID}L z(TayT+lM>3&QeJax)ngoW5y0%KEn=uZ-?D?!80kl_?&M-pRKVb0w88pp*Hf{YqXl)53n)l_0%qS;+PfkAp0`2`sTYkdO28liD zM56Z>>C5kd{UZHO^lJaN^bd$+7GF-je^)TF=0XkOvo$7&4Z*BLuem`|MDb+_1`%ZN z@bhvla7?A51p;(YjBl|QoE*br9))s6w=`CVL#(r@iRl{RK@6l$n`IB5RU}pm+lioP_%PGD@&aYOUe} z-8saI9-{;8z4BMS|23lPN5B`2RKv@+;i~ywg6!O&lD_-%Ee1WlaN`Ucy9EoBSg%q3 zw2iO@iR|X%$vtS?bwr{o+#kCrSDbKSf{G7-a%2M9GJoC*t5++FH=HaW3RaJNNxMpS zk-_GQmpi8TMt<^gTue@Y^7xea7!xr(@y{0YZ|b<9Y^xLnerZAPRmtZa`2V9gDK;_*F@MY&;6%=j5!W;zyAL7ybN+TtreW32@@lkZy~ zX2?*&@3+7{ zK|;GS27`{oqZkOGb|$^^kz`FpYDnUUEY=O5D8-2}s4KF!Bba;PMZXAV|1UW5N8+xj z@1?`3+t2V=rEq5V+65zS6X@_0CTCE?S`u;KwI(QR zXj;;kk<^Gmu3|s<|HKK_Bb=n8n7!7Oy;gXBZKLCLuAI(W!;+@KIt%Ta_{r5)2ScLr z)_|(tGL_j%{fl?0>A9s;M~-(}YW$`r)KHRG`) zNFf+c+m_wnAA?c6QOO%wKP8`3;u)^_+XFAIu{fx@}iJ|GB1 zsk-hK4JRbtmEsIH+~)C0n?D@g)9%sIVjFMqXWPqdYph3a*r9Fec~jUh9ISzlQj6$_ z3}vp98*jPUc2nG%$krR4d|W|mVK))~S-5@z;|f?Sj9q*F_>ghWg-8b|p*kPWB>1k4 zY#lCZWka@QL*YB=qb7Mq2+CJH+;Tj=_(<~S;Vmu7)dzi`mIOO%eeI9tNZIN_pVjM4 zkw=CH(hqma*Sh3u1qIqn;Koy%0k8Gbq2hnWw*=r^+U|qv`}PXMYuNcc?_8j!WXJ8j z%r=$t*>u%+CKlCQ?fg6wznGM$whKF(e#OsY!#v>*p@d;Fj@=cmt=ZNQHos@r1=3GZ zG=4G+(GU0*x;!?YllE|XXgBY3abEY%b>075yt65~#eS-sKZQN#QS;E&tsU1p9PR9I zS7+B6SJ#?GpY-coYaeRN{Y!~16*{HnNP+m)7cK3_h-b{=2jY6rqMVOhbPo_F#;i_6cMx1PIx z&ar>g(2MLiCODaSLP&Op-+Ixxfy#GIaM&HZ*Gb%}>}3mWvG`o(;Td*d?&!|?HLiv= zLcjP-_8j208kpYiWNC2X+N(nGTG4DIwI(}0DB}N=AZztRHkzzngl`NUA}m> zoRQ&K{uMn>r*7p>eyw6sl1*#80`!<7PeCy-KGH`$1nXjP`AU04?vHt!0kn2!m+(H-qBzTknbi@e{}dKJGVb;v>sVk@YchZt5CS zWD7VWW(ShtTi8%^lQ^Vuzg`o^!)eYy`t`WT@+*q+x0PJXnmD`3QXiugqSkbbf5Ph# z-E{5Tpla($S`hr=-(QwbOWr*OKBK)X%Pk6CZH^&dm!7#P3%aj3_R$~F(Edha)9Pd4 z4^w{{`pL%7;+nm z@O7G=Zwc3o+fcYgB0kU@Gfux$2C)WG1YTy@AZrw3;$=IVWEOSSBe&}`yqedF*kH-T z4lQXK14cz_@-&T4*1?Vi%4xVyQjefn&^GHzcYV#zq3#^>Z#7oXzY&W`?GRg$z9z&LV~gFr%$d~Wi&cpiaE&V6B`?W={{q8jqn#R7 z#fR7m%ehz8~$Emt->Y9-fBZKQM25H%7?o?dLOY>FfV; zKEtlpuhrTvUacwgU8PKPg*g>`h&d)CBgQ*IbsGV=@dG_D$5QBz;}AY=7PYTg8?Qe- z{8P!7<04RWZ^6>Qhcvok%rT>pW|gi`1HaaSt)bi*UUnI7z1j2f`0w|s^l!?t25xcJ zZZ)IosBR>V&>t8?LtM%_y0A`bNon@7CvczN0v=lepVAJs_|8kWG%p4=ooaEw&31j0 z&KWY~Tl=@_kCoF7%2tgE?SR!7OO6Q8d~r805L*ws&2h37`wa_rqOdZ)uvsIPWO>1Y zoR&GiiRe6i%XRveaOPHH<=+hLqxoF%J0H=k+gf0PT{P_EwQLxyTidr}TDF)v!l^bo zy7(YS7)3<-kZWuhFF);1<#m(4jKPn*Fk-}e!IlbAz2POoMK z58q!b;Kfng^!oD^wYMNas=2jm$9;2U#*ilCn8v2A$>U=qU18=J=)^!rZQOM0MHemx)#^i7^$ZfmjM8{$}Eg#`j1G&YQ4>|2?j`5)-?COtp;45dkHCboV zI@Pw|-RGO-;6%K}*_P?OZdb-BUy~TQE#c8yjqfE@38N=zfRe;sBo2`KcSS5v+PsFB9y0U6o}8|30w}i-B78y5IYzLXT&&@71pUzZ6#~b5G z8mY>g(FRvNJgT~NTmX;icslY2gKkh(3=A#bV)L>Emrj4=(CcWgLsR23Z(#I_cicSX z3c99Y+2tQHkg$4D4wH3V^O~|ws9rn>A=R}Z#6X`#NngfYj}(+`WO(!&nT%s@&86)= zuT8l)MEilPKwfz~4=oSrp_Fs2%BG>N;j3RhPUs$T>Z*g~xZOfzgJ(rHpgT=GJg^#L zVh=Z(kJzhqYL@S4TE6q)p!o8sdv9@{4?$u>HYW z9#Q%YeHAv!{#I|p&LQD%LupWiL&jJ*b>$+zFyuF8l9*o!jWvq?Tc&gUkZb*rP%zZ!_*|VE6zO>do=4Fre3|ZE`FdFN6Qnb`fAU*n2)q|pbrH3ph$MpoOv#?Se{)~njSe`l zl^I|5z7NI|mwSI4H)+Tc>7lo02Uebxr|0*V;i`AMpx{;!K4f-~7{EN2;T{-5F4Te2 z#h8&35W}#$DpywZs;-DzZ_EU ziXqqC!A3_dteKHv&kiqXxe7`_V7YS#@YppCcLs^>G#nRlw+E3;**xPq9R&ImtM;7U z^emBy2S*a!(z!_9B}$kb&6Mg#cU~QIT|-vqg>MIqvqu|#uhS2}!iw#@IOzHw@!!#? z3kL{VmuH zcQWGz_q}?Q;tdzvRf|MX0o5&TV;9@a)4mJ6?~@sDhX*AG4S+Kk+;C>bxhS8W6{>qU ztH!(amlr+#ae@n^L?& z&#A?z#gaT0jU)bA=-|T~&FEw9aPJUgYBUqhoA8oE$LuyOw`-8qiHU14O63ImNrr88 z+NwiXZNOiU`af=SmLey1EIJ;7l@SUFqUwMK7HN#}SEH%!xUGX^*~V}d)DxOqbc2L= zp=8joStIcP_u8Q8dw-nfTd1@gPtU4+_#Na*#fVLVY#?YV(sIQ_X?`xlZ4h75W!$SG zg!EkRiOgF^FJ=uf-0mQ$b}RA*vC|a?+i}+`ajZRy5~kgxz1qUngITM^BnIkb`BCti zvN}&$UJ#EyM9ySk_lFDcAQLtfg!e>x1IDIWZBlOKA-@9v!jc-2di9zVQvo!kDXK_K zPA-k(y}>(JbIqqRqQ$tEDP!Eh6vC~fH(~LhYtbOY(uC;KI~F49t$zH@mBS8_y0y zFD7OFAh(bJ4F7g>3q;5#>4^&ldQw%~6Jm_?jHdU5h(Vasp(n?BavMzj2YyH2`|y8pl754 zz2XIUNHInFaL4GA-tTUwdw{eNPN?e03wfYeO8tG$vEuuW1N(!Z<#UA+Rmo&M3fU$& z>gLAK$A;Sjt{(^Lt4%oktv`^>c~hkK0kgyXQPR4WU684_v{)d;ZK=3Au$uR6Ax>!p zIsCHK@9i^{r6fa)D~WiQiLt`B1FKh0CDn}r+3KG@m}4Ih{xbjwO9g>d)Yq?t^EQCh zOE6L_KABfOkdvX}E)%WX$Mg&VYH$rWVAkxF}m(% z{dIuVm>o3!+0>;Lzlz;d<2J8SFFH@(LAXN;qNeH0l}t|H87$pyy16$fRvfy-G0i&Q z(WL=lplk19;ZFlx1;K#2a{3jv4RB@jLaUN(^g?A2N(abg5yw&lOuvz~PF}W#GkX9y zwfJF}rdWIiUwBtZ7snvG1D})>tbaz^TlPRyVlq2 zkU1(-!g-mgHoWrKkA_aAI8R}n)pM3!GiLnAc@LO!OvOW{Er(Y$9fnj*oPSo)Nl5Zq ztky5!#;G`1=Yx4p-=tzZH_)ua;Nz4%EBoRA!#z8Ie2@upXc~J$Nx58ibBhNG$5t#J z*jidT{h`&SsaJ<84Gcjq{0W*M+m-n+oUE1e5AB5%BLEk(w_i0C?NF<6=eVZlq%z~h zVeZ?TTMI<;3YzxkJ7fvT1H3`+ROt(3!fyv4cZA)0D3nVnt>zVKS~JhPz9YtakR+JZT+ zVBlilx@MHjuhZ@4O7==&I_EMM6kZX`K82Sr^SQz+fpN_}+IuR?=XGHW^MgX3HI`C& zsdh8WcZw9v0HgF_zVl~StKoq`rW$~;dNP~Y2BwfZRPw9JmM!(#LH{oFeib6rStue*2 zkZYz`^#6}NzlTJqzXK}B1C?dTU>%-c#ypm0adcoXl!5)>XEDYSCHS zPqJF!otw>C(**^_a1)v=oH3;JUzZl|ZiDII6yckQASlx5Q6eL9g@P3s`BW%3c^CdKT&@KqKXoP~B zFibVHYRb`66emp?8AFPSd?{NYUxa9Z91|HA%7s6Q@p9v58qErM)kRJ83Kb@ho+Iu# zD`?rbf-*z~-19d*OocNi3HP4J(Vdw$U70t9EO6K=qj{9DoRAw2Xt-x@3QKO9%nsyX zg`=|R#AnmB1}r`n+${6sp1jG@;lFe}5z&c546LH7Zi}n`k+2e1{ZK(h!ah|)BVlR2 z;%48hr}^EK?ZDPaJo63BnO~4oBKe3#I%#v2rN3GBhBURNQDKm^XAh)x=S{syohzKm z$&PEST*`V=c$^r1r1iae%zYXzn1c9?^ZW+VFVHMXW5~K(q>yKYRHu>$e(0Zx zsA-*Rxq*++vyzRw8S67Wd~SUB=ZaWn(gGSm!kL7ioAKZJ$6ryzKcXyqH$i`sRnAqY z$>nE4fcP%+C$wIS%H(et~Bd!{ia>2AjOB!#EP%SKCIr4fhf%z%^N4dJT`997;7 zo{ne-DiiS)b8ZlMv845ltB-MZ2m=EA505c~$2d?ufZ-k`n8+9(@zOY{eBAVrxCn>> zJaJQRuyR21r%h%<2V`?5nCwT!L&+G56@+IJrLo4)GR7JcP;ND3?5BFO#)OiFc`z;D zBV+X8F>qBX#Rrtz3^~EynIIgYQobG>;n6CfDUYo0vfo3L6FeI4n8~2Lo)aXG zl!OJ8_Sg-&jfdpAdQs|2gd-mOhBIg5Y)B{|;HWJtb-8Ykq8!#W;=IEB*9{z}VUw7n z9o2B?2EeO^#W{y*Xu1*Iq-u1iG7tLFGAWrPhRg5jqwh9+aDyU#OA+&abk*Nzc>f0M znFrk^WQ!xf_sg<`l_^CF&93<5zGQZ+rGn-2T@zNxlOIcgVoX?BaG)i+V;0P4;S(8d zf~@UId-TppllZ`0bS}CS5}Rg~VLH%zPw5H?E7J%3*il(l;@Wuwe#+GFeMV&~og|wy zm)F>%A@ux>i&>Vpk}Ikp!Rt3>SMma!1nMoFY@j68AH0vFEh z%9`lvSmaqtdFE=J2`ALeik?KubvAOf=v9>K7T3lbIAT)tTtcp;v9d|?cumd!zj+!T z{U3QgC*}!{E5FE-)@~=@p34}6BxubouBUInTZN+MYq3s<59Zx#PhVF$wP&+pq($M( zVAhRkO-lBe%no001-M{5@MOf@)%XACdh`aO(5_XH6nA;p_1hZ_({I3$!&x`UlO=@vD1OF9j<9N;;W$~+VdqS08E{Hx1YB;oODb~wk;h?MGM!m7?MV@L92P$w5Cz6 zi6YHapSw#SRqMFqbWM=TbIMJ2!<1)ZxL&U0es!ZF{D#SFDxI!sY#nZ#$Z+Fs z@Y;^{&afM9Ci5cZAEZb^J+Bs2aOW!$B#q(6+r(-Zq%F>^uO!cj7^&O|4lOQeGL5arQ17hed*2{V-pA6qBq) zqn2Sj2)y(ItK^3JW3=6`!P2=SJ9Ku=)F0a5CN95YMvPf~2i^>r)9S5ONS{aj&?L9{ zLDQ)z5|1yBe%+*LPtaVGtD~9Ii3tz(FTC78H z&zR9Lv;Ul~mT&pw@U(w~9k~4cp}@{>`(5AmH#%a4uluWlN3o-Nu6ffj?KfZdU%vSL zKm05H(+_#vA@9y}{jM|pll}&A1LG(UD6@B->399Rf6_rnxiJ!!**d@IcXjqpI*tYD zfdd~M!uI49I?Aa%qnlq_((E31MqAbAt-sw@a=^`ws`}nLZJ5TnX88#PI}e`v#C0jh z9A7wHwJm{7Z&pFleL^u;VaYR2kvFMU$*r!ij>>ig{6%8jOVAw$;bibUmcYP-O2m2u z{P0oI4_s}(;OvJcE4Qbgns~;;=GNhKZO50L@AtdjM^VoTRs9C2 zIEI7*^}y1@^~b{@I5vLkJz0l>X58)-4_R6b3}Rc3J&%6Jz1K%msk-Vjx4K2-WeA2- z&GC_yU5T*pp&BIr`dB@x`T>akAy?lO;kdOFVSe|bi8q_#%woUFfA`|*0yDRYzO1gL zmx!18t7Ej>N->6@#PQycsk|?tG7M}9a;^Hq!vGn@ynu(%&9p8QBzPn;%WCnGHA?|D>QkC_7* z&Bd{NV3`dmY+4MhrCw@&nsW{aEnJdz!?R{E8Z^QzLeUeYkye^qa({P*?{dCS%l)?h zMcxX@ZLslMR+QW&e@9bP?`YIsEW;)BCl~lE=ZmyG;Tj>a-#Cpj1}_>Ae2{QfMQA+h zq{;oJQioH^#r1C>j2*$o|FEnVWPw#zMgwCs(-Woa5m8Ny?58sA1|PRREj9p^oC zEYcbLG*A4xjri3@)JyyrQ>~N^KZ-`S3-b8)` z$fsNH8O9+zhMVqah-R2CyzffIsEos4qACsk#;B#jR6Rh2Ar=waL!%K+5y4g3d+YKZ zl}0$xhkEqzQIAq193>c|TEn&XS+!wZvBHtQh9iB%K01^Pzg9Rp)rMJ*;zH7}Dw#%T z?lS~=+PsI+2>ZqKa!o~3pV=#n^vTwm=Z1B~RQ#h)_>l03=7ac!MOx2~YN4*r5HON| zR(&7$cS;Wt1t1p15CpfkuMp!6xXo9?)e?qfSmE)SQk5TUe0?;|Xg&}%1Pg4n;f}g+ z4skWwjC1ty+Gtx87u_ekL#Y~Le8eKyb;4T|w#N84b<(cVn(HI-7q>%91IMX-6~TQ@ z+M86a^nte_Qj7)8o<6m(T`b=XHAISVX%8t0T?dl6_L}+}T~)1syr+a3Y_hKBztw`d z4>jrO4+w6iv{l!CmX?xXZl^J+kP=p1RSRqTdgLQ@QguZwtfq8^Yd=dTDLntExi90D zV)~0};f21QyP{4LXs-HREo6%6FR8)UA}<&(sDaG@4 zwP@mB~^;n;f+SlmVXsD)Jl6#u9Q6ZmU+iNmSJ+w4uHXYI! zceBS+SZJ+Slr$C_tm;j=j*yHgcg+)UPQe(|nta@2l5%G246mv& zs^Wm2VX9WC#W@W$Regj!k}4yajLS@CRG%xVPlx~oO@_a+QpWmrSBA0nel{bxdyU^wBcQHfpc7Vn$0&-hJmJR?zB00Qr0=Y^ zQF7xb!9R}hC26j|w|DIw*_(o)pq2=IR%&#>qd15%XJdpi($e+kar1E{meg)ofknQ^ zz+ar0E(>ygG2Ay229n4ug|XYso$a+oGRLzv$xH1@?#o__8r_lH%u#+>lo{Z9&bhgh z_rv{DgzJ&E2|vLZzjAYJ_rrYxIMkC-)W?MlX52A*Zc!`r@kS ztx@dvr#N^lsi3Hu@t!ki@MkL)dW`vbg=Kt=zXfOe(uKotgaw(yBeHwC$_6D<)Sv>B8a;Ss0CbfOChDG z><6nW;MULP1Z@+jZW~Y%M9m3C(h3T}l>*}SR4oO{F2M_J>5W1VC@2@ja?S5O=cHVA z_xt;O|M~URlXK3@oSA24o_S{Gndh?hUB#ZA(gW&_815N-M2EE+gBN7yy!Epv%>g~g z^<@n1$r!RIXL`_F^5v3U_1+)ruhaI7gm-1#znNdz*pM;kP=>BztZIJ;IGat|;4*cs zw|<{@_){C-n!qrdzHcD0lKstl?#j@)<7P}dU-jM^9h8YYx*1c?TkfHzd$cpAoS*&# z(PnThrnY}^*DIyVABOF}-(cbbQ+Y}ZoFmL1Je8T|nbr*3QK0J&Gs!=scCM`3K6coq z1f3&}d*Hn7%{6!cqY^fwwvIXvK3#x;EIk73&tsiwpFh}QHNG~cg30k_;a_ry#F&zR!QA-dpM6{D&eM&!_3 zjc^rGcGZ&;u22L^}u4(9r6T7*`EpkhwAOyM5& z+^nj%wxHZVa#?Rw&4KXo0?ACO_NF>-vafNl|4l_;lxt5C1iVRn%#x*5Uhf(l#<@Pp z4`vNoQd_^S*85I9JZWxOSKlD^$gnsZyfVt{&*uw$GQ)>Hc}2edn})2!nt47++TIz^ zH?NI$hi@9#A&|i*2ZQAM;uK$Ta`xxO>bHJ!$5dY4>OaSAJl8N!%B~^xY|6QZX@vLZ zHxOAcOnnc-z_20#^M3#)w4|Q;f9V0V{{b33@UqYYqnK4sJw>WCB+5J=LbCa*0$rtZ znTOXBK(dn8twZx8mB{6M8&8`uTv>@s3QP=TB?F;c^AtcapcS^!Kr0{@>e8H&urO?` zQcQj;LT&x?Id0jxy*cax5q@Gc540VVp3h5_Ok5`O&xom97tBoZWhOF}hAA=_1WPgaa#q}Dc`i`ZpLlqK)EGKVBMFPCPf`Z9G( zDykse9!PSiYT|5_pN@0DT!1s(E@1$;LyMe)ybPJcq#GnC;EpqOscmV`J?hd0sg2)} zJ$O??38;cgpO#Q559$Z`^n=}tQde%N^=fy(1g%W#iL)0Cu6jr3QGed-(ZbmtKE zqjJ#@yEF+L6%e`!Nr|V(4g#xyRJ~*5`XAZ_SaTkBW3o;P+ zH-U!SqO=mrY&u~q&7LRd;?v)4*X#DL^BCQ#n%V0-Rn=yX%AF@Z#{Qp<{_94H@hMwAT2c1u!CGQ$r9UJ4*T#B5Gm0qx z-_ijk>w6M%oARd+!hP{m|7jCLq-ltPY3bITKOL-dlh9@r+sKODfasv4}i|45IPqi zum_DpPo&W0W2opht7NFVq5Tw+U|R8T1+T=4X}0VjjT0;(Dy6QHyAeM`h<}zrfX_Ho z1kY76G@PwKK@UN?9|&jHC+wDDhg>uSW2`HJ*P6!z4S21*64c+#mXrR%*^ixcr!^BW zlHSi!I39RRV~h4nV7>@gv|oZy*7VBI9$RjX-~RTr-PDrp59x0+|4=lA;eKJ?K|b}E z%D*BF*kBx5zXSRyrpZm@Rj39U+C&7qF{T_+hjCL2E_YF8nq85!tRS%@W3vnl;qsdO$RJYYSUPvI`ao^Rn2}6 zd0NGhp7@B&-iJoqWqVdPy;yg!7!x-Fik`pwL3*LuV9}?!WJ$}OfNi$&T&&mO3paJC zD|F{H{Op?!*!AU%S>jNXCCtm^E5L)$R}-9EeF@#nPd1>W8t+k z`N2f2ROZS^^JS!Z7dCBroD85b(E^|X9AS+IJ+L6A{K~Ek- zfpM6THj~HBcyGE#UNghO4>nFnXk$tqO}NBZHbn9ZCQN_3KKIMPo=34Qm-;0NM{Ywp zJ?m3_=-iL|`5(FSKL)vn{Cfx=OCa}7Qqfmw1;juoU$fCxOGtrhIJLF&$LZ78EO{~x z$bwZJ2?Y0{S!EL@bVrOpt`I7icXGslR*0rrh+z}NJ_=%XTkOs|V^av8-kyL->3;>& zUacsoAX{0XBZt@A-jU99e;Zhf31emGJ0=$XPKY2vvAk%08r%w#85jQ^q3`rA^xXxa z@8$$#cl2FMf8Ql6x|KP4F8r+qh(Y=d;`U*myf|fwTh_cI^!|_h2R{ZsNN%i-P`-Jj ze_TQ@2?f>RoB=j#9rC7h4OoV@^((4!b#SQdki!^L0@!kc6bg(b5wt}FG_qwS;{oPB z$tbm3zJ!$Yq{R|EFRR1Pa(BY$yEiT*?Z?cl&|AZI3N@_CqyEihiQ)1M|RQ`x%)&dg=A0C6pbJTrwltamRYg@?c1mFpOM>XKA|rq5MS7c zy)tE}l-u>W1x-l|@id^YJ|Ry?9O$s%fQg1LoymkWk{szT!O}|TV;l=Bmo+VgLyRnl!ok(FyZ4rF6JJi9tS?X~vFH#{T&%5@<&H^y71jMwePn;($YDeO(xjddg?{ z0hyN0oQYp%3Vwukc1&8XUp}+6=w1Z4OZpAg;KfrSgxqekNep3#sg)?1y_M!%bRH|I zo7eP$d&Z8^y>?GgeRG_zA(K6R24L-0V2Q>KBz1GZE;$`ON5nu#J>6xbI5Pe9sZYDt zy?bFQmFk!bt2cK22%&dy;BENoR`=8~DyUd&H;zQSyEN57`p@Y|T5ohuL>__V_^HXk zk2?npU%UR^R7^(qVQ#W~TK*0pIsd)9-pDeQyU3SFLs0<2X^teA{r2gh%jX{}Z#+cS zTh&L;GQ(#aJ&P@Z*vbvsW;9YU9jnQB#}b#4a!I&&&1Fy`UMY1@9f^QVEch}ot>=KPoXM{*hbQ~a6|Dm zxr=-uz{ED3)eG_pc;@QQ8hUP2|Klu-0tISf3SeGq_gU@>0p6H?g2dFa8%THfH{M{i zXVb0#8X`$^=St{#7+m_ke_>Mn@mcN#;#|0&o$8DUZ@j_z&sw`gn1fyhT^l^2vL@dU z#J&KF;T!ZW?pOo-3s4LDeIx?cEWiFG>z7m&s!irwaL5p5&6EotieUkAwxkKqk@w3y z`Q+jP-@iewkJ|SSdx>7RVY?vdjC>$|Gj}sRy(11sn#^wuN$~h9l;7i(Vy;5c-O}kl z#uawNF(dqU36Wx1|0O=Gzbzi;ju3OOUcX&ey8-nL>KfWrh%ww3O$-=FxKKqz`R}}l}7t< zH(lQ=tv?{+?mf%)^$J)?QH+*4If(j9N^KoPeJ`a#gI<4y(msHK(KjYUy}q2HOn3#a z_4T?d6g3Hl3$G7Ez)5LwK3#r{x>WyD*7WWyZCESOiPCHIJlRe0wXH;aEhLJsHOw1G zY?G!x;kCp|X3MFGI?GIn5kq#;Q~GP$Y!4k9Ls#J^UTUESS)>^j*+`3gf<^JT zMLFN1deIWI!lGVn(QL3V+bpqPS_XV$8FSw@-gy%s<5I{fHO_{q)3 zk7YW3W)|Y7XbFBwZTPWQ;m760k8cZpf(P-_(2bu%=kU`*zaI0?=aieNSax+m=V0PF~E5H=VjDrRhPmG=r9nq~#N6#pAScKCOC@j#)vgSJRpe zG_#G4{gNK=4L$G}t-VM`>xb_t^1b9@p58=X%*!=NFXmZHvWs~$P4bI*MJC0?yi$|$ zVxHZkx|ruO#aztunba5af+o$yyarS3#k@nN0T=UnOam|GT`*}c=IXVkIQ+($;xA?z z4rG3KATztae%_AE^s|}1vzZy!Ge02pyO17VQoW3u%}#kCH(jAmm&?D2=Dr$3xXBQ+z#z`YNW}>K`k_1V^905jjS&?cgOX7|%~%Nrx%_3h zltxk~<{ulB zb}|3w{QQyiB2lRyS!K>RlRt9H6lr0Wt|)^HP-p;CL~t9BO?najSnn7+4FYUI0^1?b z2gwi-wnKv^JujEU-wB)cHrNi`4qH%azTW$P1>3dW|08U#F9K|!84W}tHrP)|w7GY} z`=tQyHUjTA1l}{fLR)?_Xwp2nhzcVtwE7x?gZO965;oX78ukbr?PEe)d-<)s z+}7TpPY2liXVh3Kg#xE-TZpuL!#xz?74I_f{?mBfK{ymA8C%fYy-VA@%S3`0a1)(Z z@f3)KOx`*VmM_@WuYdb!(30Q1~z1PwCWzUb-fUYwh(+otzS|7dN0HD#lW&$?62!%;b)}sX~!QJ z+CCrao|B$81~doW(4;oI_8NDqAc*usQoOOo2m^5y@iE4d?6ETAGWA#4J z-yG{V#l^p1?wD>(1XCAFhMc_?OV)W1oM*$?b+MsIz5JwJZc^{uBi4gz${r`cse@4w zhpyV(i=J@8mzvnNkfG)t?1iUptVF!NEnzuX%pos>RL+ei;RrM&lXWEVLkIO?DVy87 z{{?snR$>q3FHt&ju#d2gKnmfzQhFc^XyK&o3N)^Jp?Os6glt4nrLn1W@3IEHQaDv{ z($09N-jQo_z-qI^nY-OJsMdr1{FSn+l{jSz8YKsmhJqW&<=30%b@agMA-yn!)BSne z&KifhxzAZNOkGpj{CI$_i?=?GRq#o=B|ea}@j=z$bq1f2+W5uYfo0C3n$lrvUqe#O zvVv{bccPFA*SF;tB5$b6!5<2g6g(QKDH>K{Z%T5O5*a38JiSWK$$=Mqhm3u#C})hNZv z?MQTv#ABPVHE`gEZVy^x^5b>nC3k`*t2=H;~_<4JS-D)b~w9GxKtdGL!fJbZR zBzUOsjH@ixXFd-L_0WkwN4C$fD)STZR&LVQ=Y2a{ZqIz)G}i7jSZpKfk%uL7=;)yG zk)5*P6TO-B@GtKB?=YP#5PXX7rqgQK>m)mZn?$ZKZKvo}TzqFVX>IH;(RkS$D!=S; zDt|Vv1-O>r!Z*xdhMwsiTEum4c}c?lN)k6e3!Dk3V8=!7X_VKlgUhZK@K*>frBIut z>ojg?Z&_uhwjXXED3w>UN`a4Y0!YW+E*SZ_m8eGQL_dc97Qd+3Pg*sfy zUCgeHI8;8P1&ye)dv%Vg^z-nv?nR>u3M)o>SW^0rVh0Q}rk{;I<)!yVpQfarjy}y! zKN)e;#&}Q3uw_R5Fh}}{XoNfcc=Rbi@bR?+vE!e|H4E2tm%;+t0Cp^@9=;R_U5;Fr zo~6N1_L3bi+ehk@-Wa#26@Ni+U4 zU78qnN4n!^+sv$e4{ML92(DX`C=8MjQoDXi#frPxE2L=D$8|`8_Kvw=H;zOAB29U^ z+@dKW-dTqOA@MD$VsD{+D&IY-W&u%7++dRI7M}n&Q%Dz%vQdSRNF=*{9#u$Dk;s!z zf{vT1^zKL=?aP>r&UDb;u{uHfxBBNuSqrKYh_S6CXJ*bA2W>Y#PuhxRQtifPH)|M1 zc(WT9<6eC#!k!}a55yR0e8T-eUIY%bTa6~v>*rvgvd1FuU`Ha<;M3D0-0AfwKR06- zc858i0~ze!NP?F?ZZlc$Levd|GZKS#_% zP62v{G=B}LYBg0#0Ao=MG@Ac;!+biDyNNV|L&mH(IkdSx3jK!n zF)|08!$$1J1&5>|M&t7q3oUs_dI&+xrduI)V{WvKhez+LqW6iyojZg~oZ{m_JiZ={ z6&y38zkVNme>r-8OSrqfk%+jI`;BOY$8K~Xf=LOFC%p^nKCAHxRa5Xc0IgeQ-;8b( zus}A2o-(*w1{Sp?(!xJ|WV)jC8E=Z2{}-#`_5XYSt@^Mw(en8e#;NPCoBK0cHjkp{npi`j179OuT-= z24Hpx`Vjq9EsQ-!%-hRA$-e6D4`Xi-KJqOb7KQ>%DFj+veJR3T6H(T-Qb0Cbj!Dpi|NBqdUgE-Ad=UKm{2Gx*IzvWr3K&pzn|56l4#VHk+qA{l8)`61coc4H2I zIm4v3maSIM|&fo-U!zlp_(&LS5hnK`O_E@uN^%V37v`@kfmXq5i4JL zkVehTc)eQ+Xeeq6F9A)w`&>!Aq;M`Sit3B#en(NG5Kf(oL<-e}QbzjFb8_@82d&PZ zN#_V2DcBV^$r*bt!U;vhC6`Q*5imFs?qr){~Fw)V5XHD!0&%YHs(Pr8i;5ss5ZK6LJZ}uq$r# zk9X*Hpnm071agp{*x>4O1@fknD=QS}t<0f(VlE=-MBBP-nS6}FV z{G?jIf|?kYaSPU-NFXXw+AznwhAR$#4%{noRpEN_$%zx!;68GsUhl?zJuWY<;tz{I z`~df*+m~(+;O=O0G;PA&)9Pv6g1a{NiI+dc-MzE@Mjh^-<09~$fB>IRCW70NH+?E=A6X$j7?6eg7al$1KO!hcqT(mq$ z%j~q`&Q#dj3aNxuFG&>>q*ZnrUT$vBOA}4iBQ}zqj`h)uEn3h3Jv|VGqAZ(UV$MM~n6>_=N^z5Zb z^{+y2sQM`ZiC0yRx^d?_=m)6NEh=)?xlFW^5!VP@rvHE24DEm}5ZeDg+ss8Xlq17t zl;}J34Iw+Ld(o6p>%|nQy}F-siCC5 z;Qc}D9kLeZei88mnP6dkQ6RnejjpP>^AV7IA&yKtKi!NVm1Yh|3Ylg}p$c~yu9r~e z7WCPtxax3yj;j!JN*S&Zh)cqihD(cUFfJ9~YQ`Nfp?#{)L@>D}SOk_OJoBPZhsrsK zJxPkkQwA$+0q`_ZBt&?czzTv*czRqUJ9wHe5+Xdkh`p-#?Fy0dAZ)crOz^Zp#C<$% zV_Tx>zGM$apT1$gi#{F023XAHB6~=Dvd`DD_YpTYu;4mBj{RqUXgu58|2T*p-k)Ib zq5LMBc65l zhYn)j79Z^ygL8f+37pcjnF8KpwE)EWD>VxJXcPOAtB7EKgGAz*rjOtZCOz^M5p)Q| zfGpo6(%{b}34#3~LasW2uAGSoZoGu$HdiWSK;|N#X^()WmxP?7)6KuCG#Dz0b@~4`D=YbTI4{u1QYrtEoN!ZJBmm>C0D521;Ka)c3-Ua<*gM@0> z|G8nSxs8Nrsu$ki;z<+L#dtpw#wUTfE`~c1!TVN$9Z^Gq0vfU`1Nl2f)-c>LQa{&o zq%4SJk;O_^bg^Q8mhxYWu z3OQxRKi1-M30C5EIc+!75Nl$RMt?40j4N3?43+5ZQ>k%Jrh2kd6YZGZkeHT|#Iyp5 z3)-$iXLLr})m(`bl3N4x`r_---2<2h8@v*q_qlGG(o5uaiHy=`%$C=K(CT%lz)Zse zIIS5$_O8!$0|43XGWqn`TGRcy+=;vO6Tcw!IZ83U12mHG7WU7_u@)C!GT9f%i?!`K z`}tLu_c@EIK3BEdV$NZnu$aeNRD7=aUW+-GSzs|gXi+ufs=lz8M=~$vn(woiAIeqz z!=m~!sQQXYU<}OIT(iky9>q*xUUQi(Mdm!0dBRNd7?*jX&pg&;o|J1I=Q2-rnICbP zr?|{hE#`-V=KFo-6rXC3%lxR%to5myT;^#$a~!k7r`qc>n_a3#pE;gc=Q5A>neTFW zi+rkmE_0^KY;c*gT;^<-+2}G4V`74;uYKl0F7w?!)qa<%IcR=BZ_c-vQ}yPwT=P($ zd9ceo#O0k2IrLn!K3DZGi#a{loRsTbm}|Z#S9Q>$`e&{=Ip@`@MP{AFoLHnfFhyEq zPOxMa6nPgEnSWDc9x>CLLE2VE%`ivIW`ceWx%Y#w>TBcjvDOh+QF2D%kS@Z;NN+6c z;ZqLrnGO8CK|T*Heu3Ab>+qFPepZoh(Wd7_mt;m+AF?!<&H@oFV}a^$qXoY%eh$hOQVh2nwJuPQH?J`LfAsYxruQ0hbbCQwVG# zF-f|80>bw>ldck6$)Lc&_8ADv$@`Sfzw>$tHf2RwPj)M?ZCeZm#4H%(CtJ7!cz&t_ zUmfO?LD$c{pr|u< zi+rIWSDDB1fWo;MMT9j$7)lSS_4=X)uOiB=psC_*s^Xn;%*`k-I@TSqP}N)q3)Qqk z9ks3pD`PD;0))RqgeGq^3U;m>`hq;_9=Mf-3G$LiG`8SN5-ZRQV>}}oNJpIo1pwT~ z>lu6iqDb1|gcJ#MqXZ-BBn|O2^k2gJ)BoSzVHH$ZJBfFedPDxSg80FxIL=ob=N!j_ z8#)OuHZ{h3kysO}d@=P?Sgl^Sx^hlb*|vz+rZG6-Q8;thmye;+m!3TC$XP#Sy}?$e zJ3ut?Svd)|bCkURqE?dJV#(Ajh*T;ebJ##LbXLlCyMn^G*8}0~7%?k$0r{fr0uhi! z9m|pm9g?yuEA23h@Gf03c69xGiYq$Kz{u+sW{~|f8m!%cKCX)~t*Y1c5%JT}?~twf zRH8sYdcXd}AkrWvW{{6H~sjCR6}e`L4&-k4`qVt z?M?fLn|2&o5QTL3c`)Ho5EE=sGs8r+r-Q=s5W6r7Nf#heZe9mTCoo9^F(Z>S(xwUL;HV>k;ojqX?s4wE<0IY)%VLfOv)gQi*!FCuZMv7j-s@grl8M3@e!m}1$cepwohWL+q9{xS z>V731Mk@BeJmRL9xoeqx%8*q{cgvVG9MKsSi`DR`j&oNg&0tVKPWnSFBdgq5H-tEv z-mD~H5U9;AW)Ko9gk(bV#wB4Momi7zwlCkI^TKfibV1C_su_%ObnyU3L#4aPM|32C zim$4b#qo|ql`Ty^9D*l%!QpLcI|u zr-q@54mGS_3NK8C75c);(F&a-xV{c`xr`nGb%v<^vD za(oqQ)_61ys)lT;s@B=i0t9b0%{b(%b{`XPxLmNcQC-!a|9@n@@)-ZOW8A7^kNv^g zN;0n$GmpC?^L6X`^F}8EUQ8Cc1|o)>_9z_AkmW+!qp&9?Y z9N+olB#8ETsRTO(D-w9Eg3j}`NF%Z1fp|lc3U>}>MvtYDk?&S*efF4F9HC_61_hY0 zcfRiHe;x7IevWNBisojVvzWUqjj(D*6*iTlbqB5a^3r#hC%PxMmap z=N=;O%-D1PrfTzF6tYIXJ}$qXl#UlM5>VNT#N z%@z@xZ++;8n6!>W$G}Pr{NVv5L9vATuxEM@Ay@;*mf7?sf$X85UpPU+9ZK7qL$ZmZ( zL_$Z+eoqo*#b;#czb%ND3;By_32XwmO4rBObZGIF5X&SZUgwb@yXY{ArsAmncON%` z{T*ow^q+sN-~)->PGHv?Dc!w8vf>A4>EqzVC-i9#|7j2RsQ~?Zf(RACXC@w2xU~AX ztQ1xHmY%HPbo$3V1a(Oq>+h*7&59=?Q3jWYTO4z=A2*q?tSMS9N+SwpCILZVbB%oa zUe5?Sse{Vq{n=La%i48u8h2US+OcX|`tjjmUxrH-eqIkZuZR6tx6n0dQ<%iGN6U{#DDth6c?8I;IErt*0 zO3-0^;qe~b-*T3e264`^?MDb^lfNksf$8 zC;qyLTXXm$H|)DQVkP@EfrB&mY&DZM!3NjvRZ_})n{BHJTsUD4L^!NBdcAB;&srC) zxEG2y=kU;IA~46(jkyFTVd)WsQ>0GSSv|f@0;1le1xP{UnS5MtuA!N)g^80iDm!cgccaBYOXOA5n29=<56`uRp zo})KAPN;MEldEj!(L4Li|1qiaEK=v{Z;o=`97Pd=76kJOq4-gPCXUs46#FHKU#42} z%HXU-WmdL~$y>5)wSztrsM7~>t>+`fi9^|qN8PISMci9QgZ_Myn>tD<)ojC2;Zy@! zRFnSz^S|`uN2uymM}4*bKr$F$5UDy$*;pGQv*TS{J{vf?c8JCWtbdCxe~2LpZoprR>~Wu(i!knI`Ng60&KgHIO<`CDJ`lG&&?wt1#zvX5;*U()M#0xvSk<>f%V&>O$B- zCpy2R5wXI%)EyMsjV$uC{hUkoe7DZ~GfA#mTBIvm6+8k*roso}II4Iyk$I(m-2|&_ ze+eF`$LB;+vuq`33tT~fc43<&EjXMQwnd6lXhk9A+Xwj|@3Iv$ws%|h2+VIelVWqo zAmWaVY5pX$ZKAf4>>42-!1a@6G93!84;*r5PH`ek-BCPj3x!$iY8YGIh9=v z9yv#P7rjM7>JkVUJiVtp2t@3&oWM$Knv#EnY2_l#@@|jB8Oqx9aUGCai zFK6!0DxlC}nO#fTv_|3EaMD|AXJWlUpVwz+=Mb!ACBkntsgqk#b)O82pai>u{hJV= zs>&u=ITR}$Ft3Uyry}7sVZ|o*eH;^HD&heAZSDdrqLZ+mO_5ps==%(poEJPnViLl-ah!2c~IBt;z7) z2)yZ#FGB%G!0iDu@&?+}}!Bnq)}0 z4mWKHK20}&8+x#tf3TYy3xGHHFDW6iz2a*ayNVs1mq`6KU$>1t=##J)JB3e0BRybW zK)Z~!lkDD5le9=^+r`2>0eebJ_^O83B7>D7IeVOhAc0p%P^;fLdSkFIb!6iiv@U70 z>btw6n-uLWb!aw~bO;O>rjUD{Kn#&~eJ-;On^Y!tNv$O4eT6+BmL||aGD*adIeE9b z0H&{*Zb6CNU2?2Qic{YBW_#`oL1`0ZWXuCC>AVCi-f(< zK^ntx&iaK=#o8{uwu{?>5|>Th74gG-{Ks7o^)G}?G9|&Q&2;#b>{Ycv241p1g{~Y8 zi;3GJA%7SDVOP-mSI*yMXjUeCFJaq-d{w>SIw9XRUHqCZ?tSFDVzNJCpy(IeBwyH( z@Jqye{U>CL)&3h2aG!?I&7Xy;$rs~4Du`_(*?!9 zhFNsi7q3ZeJU&ea@ z1FK?3Y?ZOz)X%iM5;4*?PkoiX9vorR&tx0CUKJYE#g77rR&In)pO2L8hPB(6kg1Em z7s17oVT`iw(CA&7(1ST52W^}hjiEKYLYy`H)3ob)n8hHYRTu=-9h7HZ_tiUV@lJ?n0;aU8SSv!z9Mx3%DxQ#rgDf z_GwM%bSHngGw4e~i`%?|D|d!2b@IO;AQ`G8RNda~d9tR~J$ahYu}=Oc1gC)9_9eBb zP9`(M&(&%Ve-YcfFVx$~pF~_L3>U6{v_-R{<_*o^FPPAYPQC}h8H(-*waU1qr>H$P zd%R@RpF&4F`F2DYp%Y5g`-(KRUT-1R8O`d-k3&Z~`R@?-8|cKh$^)m$4rpp?PNBto zvKnbXHLmtxt#(yyt;5z_f2mUPhQ?VV)YV~cIt1|Fr4wKeaepVjzca{9<^JAjcvIQZ za5#^t58N!JvNja#l89gHJNeDX?N8hq;e_RNWuRdr z@XOuFZ$$9_+`_Llo%{xbEa}HD=yU8~LRFpoy9j;h7JgNB@*afz8TiFG*Qzw3w>$YY z2za^tM5OHVdZKqm@T-T}1pnYY5Q)F1?4Qiv+(~<4QV++Zo`Y%Wq#D;7YTG;Ymz=U1 zmv^PkZC~r|#CSm!T;n`gJFoJrbGT`zYu&nv9p8uk+R2w8|2GA{q^OrquY#VM7jB|Y zQlS?+`DKVq;hyd^EK{Cr_yNQG(4MiH(A-YG6cMS|uaq^7TW#}b9l=$t73*T}7@`!$ z5R60}#0iD5TnTzd%#_6+{`^?eZM?4?qXz{rvD;A(#-2bMbzxh< ze7TM_yFR%#j`*nx6?F0iom&e!xq{B>S)KN#zBymb`E%Qf4klg zvwBjez2V$s2&zu((5SJpWrvoI9S}Jo6hRIe)1$1V~3Lfgq_qO4l(N{982N=HM*Xr{2@=Fv}O zAoObopN0e<0oRul*PkfYgyfxkB0@F+*UR>fd!p(|?W(udewJI**yl`KwQ-TM#?Bbm zpV%!eY}B!_B5dwNf>+FxXGa!JtS@}Bo`93MtO5M0#Hq?v8>1DWqlpg*NN9fw)=NNp zfpHGlXHvO?Yuzfh9^(b*sCSBW5lyLvzl>m>Jb=lXJ%0qP1gEdGVZX2+2ETt!4x=jN z)5gr0$tMj{I?dRD(~IN*VsUO?iPa$qDC=?@WAWKyb#cl=0cGU}bvfZLA~iB5UXe^W z9&O60n}OJX(mp)7U^d~d+J_HLP~GtDn{JRxi&GAJdx=jafAeKv7#8S7UP(d|-8{j^ z8?9rc(B#B}f2`E|tghgS6l{4VFIXpp zpB8Gfm{Ga2oip_OV8^VMxTYLNlXyzT5c3d%trkaijZ|4O&OsL^7Xl5hSqoR>k#x5=0kKJ+9)9LJ0uASeIN$*bS;yR&~&4D zl8;aF4}$@wd_07Q<9NUQsbIR+w`aPcrN7=o{5vj#smTms65s(A^_e-3p4m`>8f=y? z&sh`~{*wf!lQYN`7X31hi3wzTUMS+z+yGRl$!RIzg%vGG*SJYV7Cq=z6es5*KJI2j zjm#}wi;6to!T$+Rzs2QugiY|BjY)h_2mdUB-bO|2Udq0JxJxpW5zmGglNbow288B! z@bf!@J`b1MVZBF*HN;3wsIY^724U+IS0WCdGe~A%MReN5r{vpL>2^9atAn3~+yh)@ z2bLQCVam@&GNC6r_?ZZ-!Agb7%213eg;V;aaICZhsv{38?@>dmbF{MRLjGcP)m!Gu znb@g2S68-R9XeWxCFp2K5}*ign+`h+Qi0 zW^xVfWW5)sbm)~Ov7yWD{I3XpoXn$CE}u44LVVx>XQcjp5e6Acj4ud%Ddu%uUP)F$ zr`q{5`08!$lXgFYoDxqC2%T)_e?Zt;AOp&7Hjoa2N;EPg*~!8GD^59Wd;$H9HYIRf zmkD`&-_9SwceUL5_OLRV*LUsw_XztanpY;t%fd4~UaG+hAFkm8LS8u}FYNEp$$UG% z4)e!wm)U4Yu<^1pMo^d=2;iTfUVo$k*^GUHL?D3dB*I zM9=vVWEzF_?RMVR&dozEJusJT*VCg<#snIyYC@>hQM=%z?8 z3RgpkXy?^BOybU|e9;jMG+-6slvcim1uc0538;MC;S!&%RAsrtEy8+7xS{F>FUbNyU9|6DtliNfcGPbn~M(W4hB z-Azn9l-r3)JYX#1!Rx2?zu;8B4i*~1Ni@To+d|K@^Ut(%3Ha1yI6 zl6)7?GgIEBy+-XBeVsF~W6R3=jpGlR0S-QF$mi0OC{tDm>0P{w4PasF-m8HfH| znz7jQ+H3YYXVc!mjy>-T-o7r*0pELlZ_kF?B8iZmqBib=iO72UzP|Fy5l3r%Z9TFk zu4~Q_0|Iops7DU^O7f$#PY!hK2sqL9Cz7B_zRp3si3@PDItdA6EuIACPGqYv)YV5TRZ(*UWSgR|quNe7E;r;ctg)k$I(s1x5C+s+z z;5#be#cQw*GE0`c#9KJMR;wFZzl5Z-N~tB$_$%xgvQl|fEo|`F)8qvi1hoTPg26)R>bG9}OalSDBEZ z3`&9d6uXyvi4)qiX4|BU;`5!WCgLS1&mX8VN_`20vHF!|kTSjMgQ#AzH= zYs~qAvV;)D1&`RJgnbGjLgGIbLn>M-;S&l6$59~I7hAXbk2nY$IiRttk~G+PxIU?e zOLNH4M!(TCc8I(Q?~!Wn5oq3dkJ!BuNWfOVe+0kmdj$Mw8*uB72(Bj=w<3%&zKQhlSQ^aanF6I`L~7`XBm9#`f=6PxCyrR(A?$hg4~}0e z#tazoo6w9S{O?HAWGoKC|DuZaJVE?Y_(NnGBB^jNzXLK&#Bhi~P@i(F1^!pMeS0$f zzo)VEF|M{Y!!J#^r@x!l1ADvxSSoqR( zgRUSMNAaASJ@7P~Gmq=*| zXHs=B8&i`)iX*%X-)~j?6bZ?W@De;XDZs5pDiOz zeX5Eq)9po=I)PJ|kPd}@YU6)t3vx%e-ZlfPX#OB{v5h~EfObsiE_xzRAvvZp!>uAy z6Oo|6cdycr6-#K4nu>haicLPd{gVwF^X~QSn;WL!fqqj;(W4XJP*)DJ@0)8cqDqSf zqIjGsqWmz*w?xYi9d6@W+qeU5LEnSc18d_b5r#+)W&l6=4Z0=^dP{NUsDMk2B9KTjMRjYU%m$*54Ujo;SB`Hl;g?({t8y`Su4d%IG1CT2#NfY|8jo*N< zS{P6v>rIw!9_vuR&SB}-H{c{8u{Zq~3uG`W^THwp_5` zZTd#1>wXFC%4o3O35O!piEY^S4R@-YV_nY&ZxYbT3o?lDNFG9!4LnUs-*n z^}iJvCa&iRO%lp`ppAdDE%?EDF1IZ_Q_-G5I;ka;-^Ncu=)347j8D=}zmk3$-^M?T z;P=oX_B~H*R+U|aJ^c!*=xcH!N{Cb8rTjvxxnc}B1)Iu+ej(d_;}3p+#FZVtDF%!9 z6NEN=9KmQ#7>nj|%0MOH=smnvv~!5$j*DTjv7Q<^=I|tYNp}a zD6abd-iOpSib*W*Cj5gjIs0dj@!H(V{}XU+=00gPe?+Lr za0$3IkOe*&s%zyN5d9fgT8xZQf*uKzRk6^_J)f6ogck0x!(Yx6%DREBA#(?&j06>? z1Fiq~tr~6WBMrW^zeaL+3sU%B7HfnZBB_%hi6(C7lB~%zarsOaEW`9f8lo?85|DiW&+NOhP6Rp9tz>5;SCag_8~Z)~SN~e}Dt~fZkd`D{$Ht#WRe- z*IM~ETDiro!43EO(-lU&|1Nr?E7mcZi3fviwBQYc!a@Ux41dv>d_fJwwa6`;Sl-IN z+8W$g$IWfE-gOfvUT)==A@po2X$B^+IgS{#lxM*iTW+OsL z^!fqTC;Qhm?fSI8gcPm3v6Z8v-}n{e8?ttR*G*D#kMY80rA=RNnAtxk7SNLM<3fEc zd}=HAOACs!N)>i`VsnzgK{7`DZ9gaHnt`rS7lqRJ#+t=shb*^)cj9Z z*mg-Otqf7?uJNw;RZe@$V6KiJzkkXn!8P?GI ziwF|Nx93^cfI;jZ2HnB5=Df|cbsniGiL)$pYeMBM{AR~#xBO% z3_B{w!#1Dt|5IuB3uWt-+TM^=FK&UOHmrnEoJVE&OgSlERDTlcIDJ&*ccJ1IzPN=e zMvrlZklg<9S1U{TMJ?&`TVPmXsn)e8Ro7gPgC<+Ec8+pRq9?n4X+2?rvt8ez*m-|w zVdu2yitF{CMzo!SO(JyEr=71y)*-y;>9boiBGm@J&pvfSc~XQ#K!$sOaG z>T~K#_iR}N<`&@g&hLl#tM8e3zzj(Qn{fsENDICvL>FTWb#vxGw|e?~VhPs`$>#gr z@j>r(O--t2<1Qjx>xrK(dP)~7Z)xccz)XPGtxENHRK>|O?ADT1)z*&>A0UjikFOK9z@V9OnBDbA1S@Hf$sIS`y;rXzr?B zsRRDlo^-@xxf#Ey#6_&pS-rQn--W~d?e}5wcnzAc(g_3R&m4BrlYfb@y@zWZaM9UC z(C@Buq0Ynn$-`VHiut>LlRR|dFn|0o*M`?G!XL>)$A#AehlAXeaIHMlb(rrs%zcB{ zufjD1Cbar+aza8qAd#>|-#zv@5Z0|?TMjcYO@^C~CtNJoyu>l-P?r0M;NCPGc4!e^ zr<}HwJxHKfxb_0xt=C}Gf#A2k;qL#3w|9?k;<)<8R~O5&w192E_(qWx1TukC#%ZvD zv^JsK(;5m6Nz=wQwZSB9fn2!6^wEmOUKyvgO&TOZ`e>E7$WBN=eHv^GiPnIzK?ciQ zb8AUjH{{lkK)?{Ne%~`I8I$CBe|?_M`+5J7c4ueK%$%7ybIzGF=R9KMM^EH;`QyQy z3mr;J**C8_IVn}s;JsdWO(ooUO4M(kWBGl|e#86bWciP0q}R{zLxe1qoA)t!uFi0tF^%V<4;j1NM|y6C~7ZH)!8-AqBci0F0=gGR;&)_;>tKD6a*NPj4d;Ku& z+tplx3udigU2AKLVA{I-rIS0=xC|J%cnC$F-Sk+^P&&UZ_e0NImXL?u11hKl!h80G# z+vrS#7_HIb?g;tk5bx(kDNly7D7JSv&VsR%#r9s;AfzTz z5J5YU((u^u71F4|rNgvsFPvvjpGJ~;uWIf>r>} zrC(@pTK#E6Cb3Y2N#NY_?iD-RzzRD!%MM~ZYf>$H@y#+!jVKgsPObV_vudMOXgbX& z9lZ~6ytp#y+}eb5OA6-HRc(l0^q`5_OGsMe9}fz5AwzOeix z@Z{3e&I@c`9L(IQ;Pf@($L#Wvuv#pg1AGi7!r^Q$lb?JjGbw#Iran86#Mz4`3=CF8yUMq*& zlca$_UDExhE55R<53gQ|a-TFWyzlhRn9^}vap7qQniftdI2~Gi`hLA|?`iobr_1&H zP`?Lzi=out-xw>+J8h|s=v;(Zr+IkBMJ-l+je}K&pB#O0KH!5feKcHJ$t|3DI{*8p z#h8tTs;O$-IIi~+e9x_q>3u33Uu^PWp5a8duRkgED^ceg_pHdT8%KcdD?H0LIADN& zx1nmU8hScWmm5>IR9!ij`_l^btUI`}9CiEMT-hA;EQk*s;mg)P16w`gRQ~bcK|nTu zoi5GZ{g{4#4=1g>N~kQ=~MrT>hgbQaDDIBzMJd%4FO!TQ?UD(5HKuTY}g1B9rnACh1 z##humyYBP$&M-zaott$8XZ~Uy@2tvH!@9fP6Sp;4-44HHu~z>$6inB{-4U*A zp!FnhrHE4dHrRlq4o!(X0tUG_rRMd}Y{9U=xzJIpv%DIHKqutG73#u1gvijkIRZ5H{B{= zPq`_0%B8f-d+;Nw6&;wulK>9o(`#Z9BHGw79E*ti)e`pG3PEKY)#OCP=g1!*F4 zdc%GUJ6b-~{fn@zz*Znq<9yVSMgr!dCm*FXQsZ37Fn~8 zHG1Azam|4Ms8W8N{nQRu5=DRZO=72u-K8J+O7u^d2U<6HfoWm7n`T#M1xriKRFdiEo|;D zT^NQq3mgfHP9f(UeZj|?f}d%N$DRV|e&7iz%&65;Vg9MAlvUApm{y%tA&TpqpU-dV z{V-gnejw@57~viQ!G_I8h1?VH!=9Q6-7TcMkG#%r!gFI9OHiKKfKVyt{rDjP34h8} zrh-a4o^dzNG?pGRQ)c@|)@4#%J{m7v>Td0QI}8;*S7)JAZzjzqzm@YJ)t0^$7EG*J z`mO%mNY2#H`iRN4>8GFxXt*sbq@7xwMmbOcBE#cBx#gxWOP>ET=q%Eofx;-Q%yK-u z=+KDS&wjoz;ndC;%dv2A{3+0puCO%hR4IOkp6U&TCH1KW{Hjh_f?+{TKtY+qC-Gmd z7rIYMu}Vvt{e^HnXin(9n%8|Qzot{>P6gZnKE1YUOA>SXmZ`71aZwb`?ZO)O=F&@ApC^mR_g&)LlRZS*{vekj$s2tNb(#ItDFo(!)EYAEXm zY)X26dPO>WvUEIc7Rt5C%o z(kHoTK*Su$b+T^h$sMj3KDKfQ$F0jjYIEG4QPb%^WzVQPv7hV-*V&aGD5~kWy6Qo% zxV$H=J6Sp`9K0e_Qd?&+KaXNEZVqExxEP-G4x=5#u#NngZiU>X^|kvw4Z@Hf}ozn?SXXOtyw-@?avJ@HRe583u-zmeP4?q6_cjWNxQJ}K|M#nt@~=)x zOHTS<%(Jv=#nAn)Im!|il_3x>g7$^I?a7M{eNgHR7ZenE+gFs&LND$wD=CXB&F$;R z0)L||?d@;W)x?sfgyC75OStL%SHHp~)=fJp%{-~lJ}fG>l5t14RK;c&(V8jLb%m5N z9w(QT|EOQcIti{B-^^C7w7-1@B@wZ-IxI-&^hf-X6`iZOg(C_(UDC?yBT>?bciRwnuXE0v) z^(*9K1wA*$)ma(TIAyzv!=ibQKe_we6Zq)3gTq!^wPy*QMO-&LDK_(ocG08s9Lj=J zR#XqQ5!es0#BMfqn@yIt`lTBu8ZIcBdw#UgN3a;1BI@5+SJ8(z=TH!-ic1+cPQXqy zDK9@CM?b>9ZsyAoX^Od$x9*`??;hIzSIiDdS&APYZL%FX^9fKVBf*rpgHk(5>F7(j zvW+bxN@hFX{j6A=kr8FKj(i9bfv+wf<&#Ulf3> zToj^UE z4PI-4{TgkEH7oOu1OkYbg0{(AsfKbH~OWz6EwX8{Sg^w(+Nm68>q7w zW9h^OaeJmxfxdKKzf^gmw}*`_zGK!~M)vLAcA{<@)#H7bTjjKO=&2TC@q!ao`;s

_1Dr>Izi4YFJuLKpmeYnuO)T7R%G(6yoe4wvZr{8%d0yLnJ`Z>kxG|o|xmQ~T zdyMmIz%Br<+5g)g&$54rMAhbG`bYk`4&pydzTzj8{}=q zDuJ!w&om`R{4PfulRk53B~CBlt$(x!|iBR%=1Mc-(=r|ZL$gZ`gJ&X z-JwTQc?gaxnoC-}H>ypVCZc4}2&_$}nK*C99nYY&pt@LNqN?a?e{!5EBUWuZr4qFs zO~&K_y%SZm`g97#qwD)xG`i=All@%567xjSb;ZRp%^3wpTj%Lk4KDp|=hVxEImVWGLEGlA=m3!m zt@5E(*ZEd9gd*nY<_&K9ZVXQN-tN|2nu`LL$Kyq4G4H=x&tT(CE!#nG&X-N@I5kv# z*cA%yOPsjjwGyq}sj(HC3Kyvgnp^YST6XJ*L%*$r`Huhu%ju3*S5xa1-GyLwn_}h# zmE%5URU8yH+V&Q_-)h`(yI^0d4l5;m%m!Th1Knb+?uOsHPvPcZFlP$R$~bRy?@*w! zR<&YsBCXCI!fW2xnu>FB9F$%R7NK2z&`>Cdb!s#xgs?j>t~v;c#?{!aBl9$Kolk|o8koH1LsHn>SD+YOGt7~)X1p(gLSPKEgM+B3nFLmDQ zw~3Yg6lY%Fx*>zDqei!=+b#u!#Dq(xkf9JO>6e_6P0bBO@WsAh++P2b7TRpg%^jJ6tC-pF z3mL#Qt=082!nLgrR}p1`My77&l+9B+1^D~1Xr7B0%r*xyJl^WcY`vgml@tb%pA&rZ zZjvt-I@Q>(Dkqh^d2rrbmiJxI$Fp18H5Xvs;&zVf9r&HNqj$f4?9A3i_p%E*CwDOi z!*-5K!^-KmMYy0l)&_GJ4O^px#YvDyQ1dbt0JPEXn4mPw%aTjO9m9dX-KE)Phq!zN zhZ5L}fnOdL^<%iC5OCWH0rbm2VprgohtUBPJUelB#0mpRzYF|68~Dwq-}~^HhqrEe zuoasWXw?sEwY~Z-J}p9}6lQw+-E;}tCX=(?Bn}nzyDQ3Uc(SmCd5++{5$JW=-rT$) zgialNuOLRn{NNEt*VEw|pOyeD_9c-lq&2xI#>S1t9g&-Kt=zaK>7Hx-NM!boYyogp zqH*{CTAbRK7@MKz3ZICAy(z}YwN$I4yx$#zO)9d~L=nwCigfP{io>st4GZ8)0={pY z@D|UtCwQ_EVou;oZ7ivOZuDY~W`Qu}OIead~G5#NI_7UGH8FuNO2~|#t z?S>2Jod?l>X@#u)t-#I@NPfySxPI8qerQHul7R-q35vf5zMa5(*FhbhqP>TY$<4>^ zVAZ%la)Sv(lhK6*oixhDLan$~xLO(&?c>g9;YnYIZoJ#Wn|d zvw+rcy2CW(JAd4N{LYz_H9q=~lUYL@xLFIG`H?g&dS4o`m~JU=rGB_N5Z z3NQP>4P}ZctqNvaSex!fWQ zbiRPmz9e~lnX}r7&z2DJVZ8K$ucM?++7eRw5uShP4~kFyIQ5@>sZ~uDCpRN)32FFJ z{Ue}{=N@TIA${=mE5g60(BDT0s` zCh=G+AuYu7|9SEROrcEGf=kWQ=?N8aYkZ{O zc52e0R-u#-j(6!ii1X^3M`LC{z5#zfeEw5EDL`ogPaY1F;s)xcrxfRY4Ixoz(EwV+ zfOl-=QS5I>fa^Yt*~+@{sH9cpWe98|&5~kb?PwLNI6B4vncGc#AP!S;HGv{OKzC_| zNm8bYZKfzO0>^#g18=yzd=$ITjTFmP9fcRvdkUUgDUSTAuW)A}mMuT(E>Yn>VqGsB zWh{W3aQy0M?>E6t+elokcRI;0_W+VkRS-nG!Bu?ppwL@+tM6XaiYd4DwsO~4psH&0 zi~CWmE&JoxZzzNJ7e`}qeLp`cKXde+64bvaK@o7O_N5GE&vU?8M`H?oIY;GL0YDsu z&cDivc-oAkG0T0^j>=E=rCsqCFBhgfsJd>S7N4=?dhWPl`12pBDoz%3W>wK<_2mOB zkLP^+=!^*8#G`ULat7A7KT|^Kweo-aT@M|N&lOxLN9i7^{6l|y)jY9o6yDIa&n+6H z(qStK!BEvVz2$0lwAr@94l~c*YPF8uEmN5nH@jUI(RD0>-IWfdH%mi z2(VnO;)m?zaQ39|^)~jgDt6-#5^>_fL8~?r@2^73Tj2{Q`}pXf-m89FVtAnc8=7g< zX}WI=h+C#MFKad>+Hu+(eii$M-rJ-sDaEQxqen`OYExn(jqtEV_n zExAXPV@WR2QR-hQm3Bi;`q-r{`o>RMCphCs_>;(CW=7*6m_bIA2<%S;lbTWbIaUEN zI4xtBc6Bn_$|KdD$OVtAjDEPN=v!`0@fyWe8y9SM?g-yeL#;!r!RqYN@9Aegw)Ptc zA|oDVfHo)INf>8CLPl(OY0ylrCkX{dk2<+2oxMzOu!Qizj+ZwlcuRo4;i&S@zC%Zlvd zFFVqsX0r$m6AaYq0=&6YX$3gnXiVeE@D^Cx!qkGK*=5EeT>cTL_o(jq?uc1Zsi zyLmX>9T~P1+p(M{4b2)Ay4MTAd2p*K;cgDz92Lu3l7uu6HkjumjxY~yL7NL~2w^Re zjLkz=CGQf3LDv)3DHMmE8$!Qg$ofZ@H%HZ2;UA16ifIL+zY(I1(2E}7SysWYJ_HeL zYOsUlX=19a6~Q%8X(gMEA4;8DGjFkeSW4Plw>j8epl7$60EZ4r-S-B4r|NbI@JLym zGOVU1g?&p=4mL?&)ih{VNDX2BibqZM8>d{_rRzQD*R6#fO{OJc)N9eKo8vupj7Vb#0dbNeF#DcdaCOSn=Xy<|N`TXqq=-0%q^<0&G3vt3D$C`3QVL!Y(JfPZ z@GY(gzvooyLnYrNJ>F4J;dKM{3PoTUkK64}wN77&E@6$IC6q7m!_eU!KeJJ?_r)XC zIT9TFEj=QyLj7Ja)Nor2QOAx@`XsblkZzN|BIg-x-iw|6SJo>j6>|)1ehb+P998}i zyR*)>t;>d*+Co!|4ZPioE9daJLAS zhy>iSkZmnfrCGCcp3%Ibouq}I?VoW389>nBd+La6

mA-Yt~G^OxjK{C27_fn`#R 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 Date: Thu, 17 Jul 2025 16:15:10 +0200 Subject: [PATCH 127/173] feat(board): use additional button for APOTA --- variants/sensebox_eye/variant.cpp | 50 +++++++++++-------------------- 1 file changed, 17 insertions(+), 33 deletions(-) diff --git a/variants/sensebox_eye/variant.cpp b/variants/sensebox_eye/variant.cpp index c802c614cc1..1eef839494e 100644 --- a/variants/sensebox_eye/variant.cpp +++ b/variants/sensebox_eye/variant.cpp @@ -72,45 +72,29 @@ void initVariant(void) { .loop_count = 0 }; - uint8_t pixel[3] = { 0x10, 0x00, 0x00 }; // green - blinkLED(pixel, led_chan, ws2812_encoder, tx_config); - // define button pin - pinMode(0, INPUT_PULLUP); - - // keep button pressed - unsigned long pressStartTime = 0; - bool buttonPressed = false; - - // Wait 3.5 seconds for the button to be pressed - unsigned long startTime = millis(); + pinMode(47, INPUT_PULLUP); // Check if button is pressed - while (millis() - startTime < 3500) { - if (digitalRead(0) == LOW) { - if (!buttonPressed) { - // The button was pressed - buttonPressed = true; - } - } else if (buttonPressed) { - // When the button is pressed and then released, boot into the OTA1 partition - const esp_partition_t *ota1_partition = esp_partition_find_first(ESP_PARTITION_TYPE_APP, ESP_PARTITION_SUBTYPE_APP_OTA_1, NULL); + if (digitalRead(47) == LOW) { + // When the button is pressed and then released, boot into the OTA1 partition + const esp_partition_t *ota1_partition = esp_partition_find_first(ESP_PARTITION_TYPE_APP, ESP_PARTITION_SUBTYPE_APP_OTA_1, NULL); - if (ota1_partition) { - esp_err_t err = esp_ota_set_boot_partition(ota1_partition); - if (err == ESP_OK) { - uint8_t pixel[3] = { 0x00, 0x00, 0x10 }; // blue - blinkLED(pixel, led_chan, ws2812_encoder, tx_config); - esp_restart(); // restart, to boot OTA1 partition - } else { - uint8_t pixel[3] = { 0x00, 0x10, 0x00 }; // red - blinkLED(pixel, led_chan, ws2812_encoder, tx_config); - ESP_LOGE("OTA", "Error setting OTA1 partition: %s", esp_err_to_name(err)); - } + if (ota1_partition) { + esp_err_t err = esp_ota_set_boot_partition(ota1_partition); + if (err == ESP_OK) { + uint8_t pixel[3] = { 0x00, 0x00, 0x10 }; // blue + blinkLED(pixel, led_chan, ws2812_encoder, tx_config); + esp_restart(); // restart, to boot OTA1 partition + } else { + uint8_t pixel[3] = { 0x00, 0x10, 0x00 }; // red + blinkLED(pixel, led_chan, ws2812_encoder, tx_config); + ESP_LOGE("OTA", "Error setting OTA1 partition: %s", esp_err_to_name(err)); } - // Abort after releasing the button - break; } + } else { + uint8_t pixel[3] = { 0x10, 0x00, 0x00 }; // green + blinkLED(pixel, led_chan, ws2812_encoder, tx_config); } } } From cf0a58908de30271845b036c17ef42ed2436d466 Mon Sep 17 00:00:00 2001 From: Paula Scharf Date: Fri, 18 Jul 2025 13:47:24 +0200 Subject: [PATCH 128/173] feat(board): remove APOTA --- boards.txt | 2 +- variants/sensebox_eye/APOTA.bin | Bin 976464 -> 0 bytes variants/sensebox_eye/APOTA.ino | 287 ------------------------------ variants/sensebox_eye/variant.cpp | 54 +----- 4 files changed, 3 insertions(+), 340 deletions(-) delete mode 100644 variants/sensebox_eye/APOTA.bin delete mode 100644 variants/sensebox_eye/APOTA.ino diff --git a/boards.txt b/boards.txt index faa7f94a994..2363fe2a63b 100644 --- a/boards.txt +++ b/boards.txt @@ -41448,7 +41448,7 @@ sensebox_eye.menu.PartitionScheme.tinyuf2=TinyUF2 Compatibility (2MB APP/12MB FF sensebox_eye.menu.PartitionScheme.tinyuf2.build.custom_bootloader=bootloader_tinyuf2 sensebox_eye.menu.PartitionScheme.tinyuf2.build.custom_partitions=partitions_tinyuf2 sensebox_eye.menu.PartitionScheme.tinyuf2.upload.maximum_size=2097152 -sensebox_eye.menu.PartitionScheme.tinyuf2.upload.extra_flags=0x410000 "{runtime.platform.path}/variants/{build.variant}/tinyuf2.bin" 0x210000 "{runtime.platform.path}/variants/{build.variant}/APOTA.bin" +sensebox_eye.menu.PartitionScheme.tinyuf2.upload.extra_flags=0x410000 "{runtime.platform.path}/variants/{build.variant}/tinyuf2.bin" sensebox_eye.menu.PartitionScheme.gen4esp32scheme4=Huge App (16MB APP) sensebox_eye.menu.PartitionScheme.gen4esp32scheme4.build.custom_partitions=gen4esp32_16MBapp sensebox_eye.menu.PartitionScheme.gen4esp32scheme4.upload.maximum_size=16646144 diff --git a/variants/sensebox_eye/APOTA.bin b/variants/sensebox_eye/APOTA.bin deleted file mode 100644 index 571336d378a9aaee97e074142d0e73cad85eacc9..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 976464 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≷x^7FI%UcR3XkqLI_RFGL$vd@ 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%0DP+A;b81N@9EePsk-uYiI&kE zmd$=#boxb?JwY%H2#F_z{bv$%7cUII;Ysr4Z-&#eOBc;eK?5aNEZPjSSydgW zt&qfYt0P8sG&R1b@o01J`k4>2CvJddJ~8|Uzx6x53GnB4Hm;ZR#YQ`aIHVvkz(*#7qM`?03hh$szg6?)Tje(&qWb0)`Szbs>MQX3v^^e$)w z@6&L>Aj!K0Q9E#Q`p-5myGiO~&+6_6HliGXq@^t43Ck2>c&$UxtA+ zDi`YTJh@i|7uYhLt5QJ0Rf#nR{&JSNCaG#VheumK{gCaDF`0xiq0u`@LP?xx&X1F{ zPfw982$(_YO(g39P+69PE2(c*^NR;R^Xkua||$)i(^4P14^J*9gkAqfae zBV;H|P2lvC`Dh2<#0phE_)$;ls`Y*!4DOEMq9~_rsz`r&CHtbONwf zN4SlB=l#|K%*N1?FMa%^=(8LFRFtMJ0<`rgA>B}#8i&9NN`alC6Tav*C>EbEEE+v) zuDm)zUIMpv#>|g=(T8HXoJc9bUEz4lLI=Sg^%?uV@|S{Yso(Q)6H<8#Qs9=LG&OR< z=m~PdbrWO4DNa(H&cyeCE#HT_j)QQy(}c%Z|_i*SQM4FbSB5V67r{<>~2CW z?WGe0s>Tye??R{eg!Ac_*bjCjvE35nJgP<*!J_Q%=p8&?w;mQ!XuZv$GVFO&gB2S5 z*fbSo;G5HfI2gvPQ9_X6WD0*#QeTKtGwy`7gD+zT z?B55iu}f0PMyFdVMmyQ36w0oG8}Y~Xhepq4J+ybN(!JTdCNec~mTpm+IMPYi$D;(- zp?A81?XSXLaFQlho?dIH39bxtx`I{&xiiHeC+iE;m8j5w4rXC6>Wb~Vo05*y^HDa1 z`AWL^m5lNghMMk*jT_4Q66k$*FQpUjWYXKpDYeImYQNY!F#4bJkvm&uIHr*@PL&0R09rYRzM; zJH!ImX>U$~B{symN0_ZiQTylg2j2e=(STIG)8J>0=S9R<_B4pyA|OcwfpUPb9NEUlS>Y}!W1G+w~*w(%XYWAW4@;#j50tk2Tv_2*)SM>!Wtwl5iyJ(& zm!SMI0mTQ(z97xTc@6A5efZ19KK)ojT!b;sWCVlVRrtFGljg*vPW!WvcUn$c4x8ie zZ8xb|`{Q$@_(v$7O>dZ}b8GRsfOc$xJ71hiaS`L-<{+1A`!5b$ulHWB7q8dPeTn^%2DtKSefNgYG6)_TCfo5OHrZ|+ zC5&||t>dI7;cis;Lp>jf*A;0=HHC)`oUiwuukRFBi|6avQ-q`NyLu=|L-T>np;|PB zj~Mu--uo3`ULalygdOUSba;Xxto9Xk4IxI|3J2@E!i4bBC@md8|DlvFqYz&j1sP%X zZz(tl?-Ek*)7$b%G#KX`Aqn~d4_KF#AJ&T>)V&3*<=D^Pq4L>V_0}dx zA}*%}bR4KhHQ%E6jyLPw;Rw7b1-k0p@d)gtKv%e9IhnL(yQNS(Fug8)O~F?OeMz=g zu(!0;^Kxk9`1>4fT0+&MI)PnOZK%poRe6)Da)M^X@YY4iD>7<2MvJVT5FC&bYu@=} z@Oow7PCzDOJ`8{Sb;@7B1YY{4HbRvunt zE;Ao|LvHDlZB3EYRzdioxxP7t*G>$wYOLYCD$W|UC^&Ib+K%`xSyauE)b5|Uz<51x z>h{PXvz^|H5BPCbWpQ)&^2=d~vihN5&k)=X# z!2GDuyfe8huePXkw=Am28N{{k>2BIB3*K*U>~;oYcA^ZAygUr2@XfhkssAcI1#%*6 zJE5iMK@HcubbWCAZu#mXvZ(l;`mYXe-K0SqHp$z&x|`6J{gULR-QCSvXpyj8^qZ^X zilz?isouXl?Fi}aqbx`L8GEzE?y&H36p2Bic^Kla2fLTXxN<_h5M&!&mRLJgVm|v} zn2FN-#}nldW9!*}Xh&w_>O~{%RG^)t|5Ppej1v6$#Hf09T!M?N7b6G`@;cIgik5wF z3vN)ju|61vU)dd04hyco*-v@hE3|uw0>J!){&FxBhl2=)zVHMc`^t?~!RvL5qqs+d zp;&~O&8|7pw{}0dM-Ce~eb9`4tO#No-9>L=ZWXNSZf@GwRkTkQwI9r2KVHy>Kl$my z_44SY@TnUfXNod~yEJ)W6Yv%{n#KXvV6}GBhHhhMrV1XXjP{A_3~fNGbnG9r)|*SQof)qp#RpYG=o1j-itd?`IB>VgU`ik*QBxr~-glg>uWcmTwjmM}EKYSxKlAj$n5-+T*iOO(m`VV7eA0P<6Q;mIp4TxD%vIRat z4sMH`%hQe(dmMDA-b-~5G78QmGH6=yO$ST7H3r0guv4?3^$u^nLu_z# z)|=SZR9hE>aeK(XBjzcEVfu)RUh1cNw&74`O3KV?9P9DqklhKY#=*EiJq3AD#CIm6 zq#%zvWsj5%R5-j94zbeF33k={gs)gp_mvRu(pVKRC<2jy$nq3dIO5S;g_N!a@dZnb zf1>X||2yyo{J#_?Empp#<*>L$>bB2m9f?H_KEkFePml;n^%|X6=&)*1!)xN%CzMW0 z6mqm+pFAZ0kjL?@gg_Zbu9kk|gGCw2djdh1X=jzx}e>a{> z?LA_pk;;WW=yj)BtNd3y6ZR$n3)W}Ui*KOB- zs!4RKGZna1Ro|s7j_Idnq5C?;gBG6xlA}>@`U>; zpUp*P_5qEupGdx!GFF}vkK|NTL3M(i6m6+_qmy8|1v^@z$!6?qdkg*UvbMv;eH6aj zR>OXh58peBqEo@ir^wWe&Xa09?kSpG>>{CL`>71{TLgtnemk-_XN=M1vso`Sz+oJN z&$s=bcf)Q1>-K*~(f0xWcO-r9^M8lacbES=g1#k=g2Ol(XkMdemxXoGKaR+D`VWki zVX3*WNi>w0NiwC>?8@Jo$i+T+C9#Cw_)CT2t;Mp+M|T)3_6ot?6~XiEufH*^jLo4* zdoz>6q(v_0<2Q5JC|CM$t`vW+!|XvRSDv{WMlg%aj(A#&AYLEs^ z);*{g=KJnRMs|eYJ4!{1`0t^YMei#jP610mdwH-k1%QV66*ANf5U{L}=)OG!k6Re+|2cW$dij#2c0p~_M)%M6 z(-vHhc3hX>FhxKYVj2xwUgTgw!9})AEGVkzCLwgG+y=rUs2$pIwuQ|#@OA?g_-G<5 zclr~cAbY_l+1aGCO+@8jX!Ve5Z<7r=0$pd@yX=ti5|Wy`-N9YQ_rNxU_9NX5>!Dtv z&i(xn%Ss7V$HVB4q@fw2H$(Jy26_ed62%na)^ptMhsQAKTy)L_dnw^A-Q|?PBK^+RA&-sbD3_h{)LN_)xyE?t&*z1JURk_-0g4$4U z>=InL>po$hNVv8F*B^%(?cOH)u$P;`7qNbs)qW(eEDsAv?ca9!VEtq`DqZ*U(GT8O zcP+U3=-ZZ~|8nfv3k%AJ-Jk5QCIXao*P*1~+8(=)53Zi}0xPo5UVtrNaaPwj`(z82 zw(u0zWl~WO)O89PUQdUtv|n}RDJe}XvBTL$cM*yI7Tij(jKABg8s?(bbroh${ejt~hCYpkzBKelyZ3+X{w0^$DXm{7&5dda z=5fHVp%r%TyxYfaF!#2x%Xrw{y*hN`^4WQv-U#%~Gg9B?%=boxVrz%6@atr9Cc}P%*t-+H`GlJnm0U65Jk55*|O@LY$;fWJ{@- zOx(lN6t*Xs#Pc09fs9c#PDB*$E6tzgk zglzzUcq5L05ZmQvj&aHO71kVFj;x!l^8{t5cNCJ>{ml0p zpOZ5O<%b0f<%wM8-N}zwDF|@u4cws4T56F7`Z&gsINI#Zc&vX}?Q83PJ``FbW3N)S zzEGR2dqHT>xNxL%dB)#+kwa?Y-!J)T%OBz>|AMsL0UnkFIH#N?ksq+g4W#=I1Yta zek9IAxS_(*qfyQ}QaW4kXi*?`4|bB4Y**;Ursf9LL4Dwg1&}QbZkh03ljf4K4+(_5 zvUAu_;c}u-B=$jsie83~k!!-L%ucUUB(kf1?W86z-g^ zd5(lAb{Ey6f3{K7fN)G)ofr?68ttr0%kt2=C17+orK)G|Tz56m)MezO(WGNj`LLrq zlz6ysDWE0bN;4Qkq55%n!&l!%ohYZaM}4&-YM^kZw{WLexU|K+3r98 zNrof+9SoL{Jwcc|LUvjj?;V=F(|c#2`*FhD6A^A1a6@=(l9Z%iPu2`W`!{z_zW5;)#S{k}CMNnj8 z3jaoe`xI~shPa*H%K>Dt!{or+%?a*|Q9DeL%uL;q7rE15si+*-0XiEp?itWVP_*1P z-#ei%2ejr%C467*Ah>h25b(-|mlF2Y4%!DGXAW)&_Jn-A!eLJ!Ly%21093Ksw1Dt`MX$HF#T`Jq+_L}aR#n2gl(&x%xm!(ZW)m2z)teDYlQ#cYL zn1u?(qW?WPt_;v@X9=#uiFU0U>fhn5-{B_+qJ&fSSy;WJV6|P)pubK^@HK#6 zKh(X$yL-4Fu(r1B@bA54M$AA+ln|rQ1eg_e8xjV1pVcSIY46NT!nn9os-}O0iG^jgOW2ID7 z^p38M*(ol?@Pv{(C&3y7dzKQr3Q3p=oHo-o~R9VLm~b zINxPT{PWD!3=6$x!#_KHFp_xQ*|MJ7F zyKoco)pBA$nSvlb9KdyWyGJf<;HHQ;xoX{&8nB}U0gZywx$ATufPZ|+w_c9@=|kFc zc)4Dq5)W)gc!0w54s4Ivzul$Emv4eMiRB4uFpM%G%G#%$({-@=55ak-c8Bh(1)UNN zwQ~4XMT?Qkty)1WTyfWS<|QmPD9iIvWF>2(SQ>P1rE7adtsR6AWP+_h*;!ye4xaNt zT8O>j-}l%9G~&=3tJxH+KJDu(Iw&0Ot3D{a8`Yebwg=QB(HwlLBPOK6_%I)|B4$l8 z9^Ey(K0aRGA+MQmnEj0o0hnoE=dMC(05x_gNECCc*krxarEIe5T&gB3tPeqD0{bi! zwz4Cksv&98K^OxzOLf*rn?e{_v)H;4GjmkUT5Fk2CEPVDZPDzST5EG^B4*)8tHL_U zsLSipd^)m{$!!C?IK6WYoM#VejB}Oi11$q=`{W_^nmv7HlzCA_dJ!W$$ zqb8qvE!0oCqoU>eY$`YC75kc?RQL--4v8Mouh!(A@UcGydguFU6q9*~?hLLc$&pHl z*dzqGikbwatGY=$3ba7Bz;(>q4$qFcA7c?h!A=RS{27p1f>x)BvOFu6s>i@4J3t0& z(0Z9rSl%ZnT$&|WMnb(k^4(aK4(dO=2J7g|>|d5!1FA%y=3{jf zADyoTqvR%R_3g0OePid*eMleaV-=KQvFq{r9pJ0n$u`~&&2jy&u~I)y((jZs^uWA- zl%m*w>ZElkTJY2Vy^?f$yYMePiZhIT-x=b%nn`k2i*91EKwN@hBn5T(e2J9Q z`v83tPL#LL7p=|Q@8cW|xqisrBuY*9Be@syBXv|>yDvJcqyy!N&(=|W6M=oKxRkz2 zOwj3(KH>=rl>xjBzGxxS342aYN$^HA^eA3Q3Eseo*@QH6n=iUAvktP&mwRh{d~rz) zIyut@lLzW*A8+d2;^P;Vyz1lgGdBaKtGCj}+e#{YyesqX2p{X+4HKGdJR` zE2Y=!i;gG}e9@-NQiK=wzT%6{FL~J)-IcikVOuZrMY~GOzUX6_e?|CA?|K??tf%$T z+IxE|#{K5ndzHOKVXV1pzPyog^<*w@tXzE`m;Lw+7F7qqdu$;P0M|}A8&pvVl_`Ea zWXG&4Eu-Z8c*(Xu&rgM$uW)cw#n70 zkd#>n-YSN4Qy0-iou{Pcd$F9`FzZOmat$cWV-oQ+EKI( z_p65~-+DP_S<5FK&r><?bQcTK% zWjw5--IGEGp55kscAJ>H4N53%29=pNIfDIun?!8##wu-y@%?n(RN<#Z^Bx|>vbLrF zPU%mUPtOX;q_+Teib{K0kV?a&L;&}r+q}QuCjJh%7qf|k`{{DbYX#dW)h!}P2({vt z=BF>IE*aarGq#D-0l$>RNbN@=aIx|J`q(1kf?zXfq zgbpNa^FDyYQ1RYvEQHco)V6g$qT%`4jU^r}WTKLY7Ozr=WUnE1W$b+4GdkN=kLFYm z7pIa%HQV%YLHJ9by-Ys?UMccaFe*RzdjRF>?ies^^WF{Y<4Z`N4g|)C>MiM2aq76y z_g-HE^6ycofaK}!=0C;vMV~8~>8Y%K2dkFa+~ptLyjpee4BnY3<7n3y@)qJfchn|mEk_UT+_h&pde_E4_NY?^e za$e*ywImxDa~uXAF@_x<*NPw4;v!l;SfYhNs1IugKCJEdsJ5qJDw{^L7Sn**2=9A= z`eNMwS&JuQtu3WGOesX*mLZtPk1?b|RvX_Bg(NN7p!tkadWr{r@!!rE)t3XKWJ%?jW)Hx*TkbPBljqnccl@a#MvIA7J+*Wulmf~~w zqC7uD7a{qJ9|_tlR@df|tO<32gtO`<&c*8egwy#t;f(#8>`2bHY7#$_a67Aj`+75^ zuH`g}@HCP9W`5->wQiFuXiZjBvpcBrFSYKJuF>Ve)~LPEe%c$+6=WUVA8ZX05T{+MdQdu21) zB+UW;ypG*+YhDljAK|2Ft>b?u;hZyll^^FzgcCFF7EXUm72*8iHk>q0gMXLC2|FKE z{E+WNnkyMYPkP#cQ5yW)ZJ72;_1#361_;xw@!lxm@yMyY@A^(OyOObZbXamJcwD{> zoAuY&1_i?Qk6+a*x5VFv{))wR`%WBoC1bSV;Sftw(Ppz5Ki~6aK^IR zaSrbLFPySlICKB{Yn-`%A)K%N3g_TUx8roa;m3K;t$NCD;f!7TYn-ub2@K{-L%5~O>qmt)H}&o?T;7pgLG z)4dV&k#i@n)enM=X7ol|>^^HKJRi!xRq`X|mcw%Po z5yHqdBVqDbYYi6^{n|AO0aie;kBzm+<>0w1{eC?lAX53@W>n5KgdbZF`EX08awozf z0M=HFoZY`Ue>&0xZNfbKS#qXM_ntp-saV0^m3=UjO{C0KF;gW7G8~3o{$4_Wg1@l5 z^YnbQGiOogBH0D+gwtLV+B$*Ssvi7IR=zl2-wOsb=gNVfx_73T6znO=*@d9vopo#> z1rF#eUMQN@k*7!YIK^1(Z(907B8xlibx1PrH}}{LMBB~vCzc=309IsKfnZq&asyC8 z@URr=4MAQHz>t|eL{aZ+xYZiP@xnLXmr8qwzrL{(3UeDZ@TEE0R$V&U zB+T3RaekKmONH$LUbjitc_^g)l?@fn4S1mMUl>(W@ydpBSH2#ihWX*^)oPdTf+n(4 z6LH$^n_KQtzn`Oa-XTAhlfCa}L6aZepRz)c$#E+p>f)?wVXOytG3!{CN%bWkn%0q! z)WZN`{4Dkp)LmKZ?W$t`IHuA#T9p>>eMawjYtD%x_GT4gY&2P652i+5GM!`pr~)cJ z$kOw4&FM)P!;Dn;&Hd#VYCuSor(tP)=M(0+GBAcc(-vyk*OaphK}XLz_7w$wF@~op zCNPHiw~yfxJSKkXPb@#6md5boK=2o1_`VeBA47oA7`{hQ@2k0`YK&pahd0OY4UFMY zO2Zf?I#d|L5uK{YJr8Ics?qzJo8rP3-UTVx)AHXQ%juCn*eRcO0aFupm1M5dRq}_x zkw0DHl>~nACSs92k<|V|diM*HWRGP{x6bQ-mWQL$(2^i}cydTjTvl|^&zjTT2`AyK z{7~WE9A~?%E2pmz)Kuau$hl>W>Lpp*7+I3}1#3qEeg@bxRkOU*`C6d|e^ohA!9tb& zuNA^Fv-x7XE!14oZLYB|8sg8aC~U6@wHymwzE|EGL#vBfD=j@CkE+QUw)OLr`D9CH z0Womb>$2ZKr8K2!GcmRF#Y+8G5d|+Vvx?>!-LNoFS@sGvYhj9b=(2ds+tpEfX6^)! zVz5XSwP2GD-g5P`)s@q2)sb?t{E}`fQR9rYh<|TN+2@+PuCKH<))8|@?t^lBnE{q0 zD<{&_al}MX8(=+6uAYnaPhIX&I3JO1(&?{WhmwVBKD&pi))X#>TW^+NgU~T#QFjs` z)WBEX2{$E)!5WO_MP+}W7DRJVcl70=u4NYQT!(kTc5ReU8Yx##gq!90 zv!bmlgn?sT%d9c9naa94T=V(ApY}dGw>CT&0mD&HT-~AIsx-ea--!q!M&ua(KzCzbxU2AmldJPGF9vD!NgjRV(^&I(Mel#|N2@YI`7 zI4ft7xPnPjCOq~dVka&R5|-N-RAV#8!FyEnbj=uRNkE2;dLaQu=TsRY(hiT<>N^HF(a3!tVP(^(A>o0yJjTnwTuv{){nqUad^ z{pE9x?WkNYrNW{r?g6E^two+5h;=Fw6|Yrvk;{ zrUjD&NLq*&gzUROiiozWmWXcc&!MNm7|r{od5egGobBnzrX+M#cP=Pe9m*8^W4wvIp@bKqRXPeO=q1A zgEbZiMY9qnVx$BGMd&gFhhM5>OIuKfwN--??@Nq-q5c|r(NWpb9W&>~4Jdkd!f_k2%R^8LfiezSg|n|B`S{=D0as^`g5gHt6m zI90|}0>W;N9{4j&kq$u&9f@rtbd-%rC1s9ZEoVZA#4Ig6(t5ajSu82N#D)7l$ zcOSddjMg0l=(pMvn(dV~^T1dB^_MnE=I+Xa`j)uY^PrPJs8%>1dG1skR3n~*=0s?l zs642LFT^;3NL5V6-2Yee3Nx_sxi7?3%ncA+tHS&Tv7TeNBB<(njCOkW?3iLX#VC)p zKA)58^3RN%Nqi%)23CS;qu-#9M`)|tuxFe$ADMfQ`!v}qOPv1oA8eE=v6n-oEi3}g z`Z?HGf>V^DHvmbm%g@)-djEkA#P`JO8BVf3NDI&dm&~WSrTsE>^S6Z#Wj=er;o3ilXpLm_Z$CIg&7EH>I6~a%h*Ixi?zlCKMLv z+TD^Hivzjwq$xE=!zBbY=z|YVlHxpHpmRP>KzmRMf(JNdYwLqT1Df53{M?*0TAIJ8 zxI~0T{w1}`vB5C&%KG|^&88Lb?Hwsy`?W$3TGHZFP@bo1i>b%Q>2*34KKaH}beytz!Cv-)aM-Mbt$>cUA>7Jyy zZ_{EgH6r&qpYwbh#Wt5?eLnH9%a3WRDe+>_uwPLN99mw^8vAROOY~z=V7i5Xa?nhz zr!|2fJDBEzvF#~8uoY%q6P4q-EA9P%HSE12_IDe~uXsZOolmeOk13(;n4`2 zyJAKY9kHoG((?S`Pa+Q>LAyp99I?YLMPS6h!?!KQ?p+p#g!9N%u{>JCUA7TFuQIB+ zSe^=6#bgB}UiwmsOQ{GvaYBuD@RcJg6>w0hzmVZ7;%|ElEj`}aG+k7VLZ(7$ zF!%97`uba#h8oOSI0$UP{exY3?!TT#;QDwSV{=`8EJHyTSb~^`Cjv{&y|;4U5x-%H zy1#OEjCJHlj+cG)9ICY6m?d|!ub)GuL(`v^51wWV{e=zJAiM=GWQoIsdLDC5#{JRf zu&o1;SN9T)Z+j?QHPjwze6U~RbjxQJ%Q45xkY#QkRp|)D2c5{jVF#%~WIfcJmq%hb z2ll1KUnS6WD!%MXG)hW0A2|+NS6tz-t6sHTc{_kxJKAxwf(6##>}lE2i_w4*H#n|d z)d5UCN$-f;G{qUSX(m);MhGBAmPDNjb_jCdWxs7$J?Xol3-k0&X}jhxHtG#|Dr4WZ zIY$HpGK3$y+&w-M*Qs&~VuWiXq`%HNTTOaiv}ey0wv0-{pXG#qtFHbi z`b1R4U_x^1G(QU-ey+<%gA3Qs-iUzC5Wht8aMOAvq88~uPHLpMX9_g(=7CdwanCec zrF{>acC5~fq8Y_W}yrzfD5^z)#nyAw?QEIUOj$P?yCbRF-3e6#76|Q;HKY zFq^)%g(0Tb8mi?HHAD}E%>XMWCIIFk8~%b=tSA|)fM#~!H<)F32CM&dns?PyA~?V8 zs$tl-s!l*H^s^?7FaAwq?7B!H7jx-=txFf-oib2|Npb@|K};~Zv}qqDP@FEBi!+Dd z+d2H~k!==U0(Am^03N#G8&w#b3JzKF^#28S18-9R+t`t&%KvBmK{(X<`x|f~>ZL22 zS`cWMq7t=%79E%1?WLyn$6x8vMtC#6Ri6kMJF@MWo9Gzqk3erU1^ZyjAa3)GHevb# z{gyD^maF4Wgbcg_sN;H3d7Hhz8_@*&|5g~Qn4MyyT$nfY1<{nL9O|_|4Q(&}Ck-Lg zxeJHDw)1MRnUy?u6D=-<{m~#rTyeZ<34788 zmTJqg71-cNWm_pO(b-zZAT|Z)P08I<sODM< zGH>WmllHWtpkh(Ql8TiT4Hc~w9TjIPt`skcC|(kqnfjz2>@rxKw$1)^`b~QZ3)GHz zhFHalSf>`}X0#)B#k>PyD{>vh@lIzkN$OJ7rgH0sSWhn)R~ACH4I)EKA?>)J1&@y_ zu|L6Sly2CWnz`TTVj@QxPf_2wy#hOA3OyRB;}PyQPkEU6NOkv!o_N8w0q_Fqn)8PF z>iy<+4K}390 z4!ZYlVCQf84b{=_lv9jPsvgmAy5JZ#a-THDniU=IZ65>AOw0Vp?B->27A7p55-?PH zaO2t-7#xhFq2oGxn;I?|NMjw;R%=!mps&_YS<5n9J+Wm-$?+57rOYH-GA>|%Cd{)1 zO1St)Dwj9s@SDmlMg1v4SbeAh9&W~PAtOD!``&HcIh$V$Io5SxXS}mz(*R7O;3k7H z{gwxUz{63iJesu-0c8GNhyS&jdhS=o&xJtZcGR~VF~|)qp{J8^qgo?`)%@K1rcwiU zOod|U>n>3*WH_{kktG#(vJCqq206%=4&kfuO*VbIJyRNTLtr)h#xFX$_>zpSR-zpa0R#hC~>7B-&ARJZ2 zQ8XMhKC`+;HhFN;zC&ZeWdgUtoWE|KJeK$g4d-W(nKZrTe5r0~&1qYPT~64~YOX+& z=KK|YkC$fbciVDAZIqZ3h1>7#ZclPlce11(dy7=GU*Tas zS%g@2t5|lMEdxwe$ya`27*WoVUUOtCJkleCuA05rYFZp&HH|EdT0V5c1EcYQWX?BF_$$C~Wvy z(v+rXXPyU#cEPWRpH>g9fv;21Ra>jV(U8%Ne73y9jlD5gqOfUFNo+SuuBT1>0a3iO zjFO7?8~RhgjCK(28O*;I!}58-AHLdlJ2&mi;an>d%^Z3L9bC!YL}7|e#6S^>|K5*r;`_i z)+;X&v1FP+`9amLP|uzi#x?FuOKPq_kB5$NsFKEnJ1#DW{WgC3BWgHG70bfnzJy(f z#>@5*Cs?@SO8>uiX*};u8`$M9Jad_Lu#$8B!h*|ojmqO%HD{(}M%K3Y`7_@78%BbP zy~~RgT->FB9<}XT=rgW^PY#YX1B<$}oK=H;k5uGd_eF3!;(+p#`;D_7#|^LC3SX!W1($)>Q3Q-UU?)s zTpEwgU3oQp+BjXtt6jLYdxrLzY|At{2f@&Q?=K3KP9^E)I4&lI&7vVpwr1Yko{_cJ?BAIOr-RGK}&Gjz>A== zaLI)uOOb}wYwPrVZCpc7as1CW=5kqF(&5tL zR<3SQv)icaYE|>(80-#Q_oiYRde($QJ`l7-lL8s3^Cl>YafKC^P%+;Xe3E8qpu7O* z@g10?T_M;3uN9f#B`R=*P`4lR1<~UcB#6<82==)k@miK(kS=Dx$8@wpI+ayg)~%{f zf~ec4QMA3_FV?7*YzW_;@%rde!`uK4D18n4uCd*XG(CdWz1XK%eTNP1C$LppJv)os1W~E(JJ5>`WzF{dpJ*YypAa zCNFe(KO%-^#i>3hZ26h=IqYZ4gm7C9b2kRUcj(rIgDkswxFdaGWbfXHo`$;S9d#9n z+oXhf%B6uBf1n&@(S|z1>41)c*{LNSu26w`qgB;csbJ4jYaEH?VOW!8nV&e$BR5EM z>C=ucnqPd>IoJNCvh}@Px}F^y40mbzBLm+WL%ZJp>$Zjs2g+MtXl~eGkks9fnIZCC z!}L3xbNA8l&O^@6oj$uc!fuX46!G+&vAH?p=H=YO-Xh-R$oO5FUU@90@8O*LP1SxJ z6Hn?lcdMYJmgZU~tNK3j$Jh7DDVWZ^F`_wyiM)>&MLOI`W5}Vi-Ar>G$H*#rA(>x- zh;2*H+XJqVBFsWdcu?UmA1`P(6I96&|4U$^b&{FHSp|&MN@pd~keJz1RQ!n9QQ5Qz zwt^*+Z9I4o>+n!ftH^NtR*KwI1Wn01(BD?Ve9ZnxJw?#hu6W-6tN5nUbtSuDEtXSL zc~b$*5epnCcnIs@rR?5=e-+{_Yjb+ut$RnOJOKE=9gl0eG+VNcx_5Sp^`$$?$0KN} zLV*X*r=zIH<=^Z$@^`>L3RLspzTPFNPR5sg{ z&*%Ol(vYPumdA1nmWC&VQ1Ni^h5K}53{9clIld#UXJYk8Ro|To@7ykV8kmis~{5-nngmR!Gloj+rWqMs=E76ut6n z99SwALAcMRjum>|d$dcQy;ZxnlHEg?5A5i#jGx~#p=V!p9()_zy7dp}*_RD8)wz+B z)N&)(sRn=w6A33`>Ap|@fAWvx>B7lA>^{=)zxRRpmwoVV>VI?rw!H`R$M-?TI7KYs z*<*nY$TPvuf;vD&(}4Z$21x;17oGYtk;S{DvdRP#vL=3kZ7zCbCHfTDPy#ddhct^S|r_uS-imZp@ zS{6+LfCYABTKW^fE%0+4k(a`plUU`AuES`;O1M+Op5_A@W-p0AmUTJ7n}dyB z%hm*V^1>Z=IW$=L^~nmykELO%s*>C6BND*6%}GEjD09VI8MgLbdH#};a3whxJcAa-rnsh~J2*pLlVidx9 zmE*dUV~gdK_DFP=`@B5+Tg_p4;?~fT&3-rtG+gL*XWXmO9Dv(ViQR8#IKN_&W{RTl z=uh?$i9}f8bET5FR)f(z^5Ht%{~a)Frq8;|S>< zK+_N`uS!k<`P!M8RBYaTY=v(w{YJz#{i`10f)A9j1E14m}}^Y z!HV;6StZ#1iB$YP@CVTj&4GnNRLzCH1Vw3VBxYZz$?7-kxSlzw9vQk7SeIkzAxq^w zw(|Netw$bfTA{#7KyXMvuK`b~ojD2S=jX6*3`ibyhI#Lg+?&xOX=)hffxP@E`g&+cgh z8YmCG^Qe{SYPcnHo!&cHEeH06Pspl{ug0xDbXNS%mYv^Gtxogyx$H9nf?uPid#=Nr zl=}HDEiR`)yI2o9X3dqpE~HD!%k@*G15f$u$;+Rs0u;m#y8?2FTugUEWE*VjXfDlTA-jpPLXrn}aJl-iudd+@GLvkHFtgJ6@ zF8)NN`53>_M{B|ABBT7~;ac&{b2c@~N@~07XJ|5_Y%@f)&bmEHoDo(3Z}c!^;d)A3 zsni_8y&Vq2Zn*eod`U|}j{pP!ek%rzvGsOdr6&p079rf7U`T7r%x9Og{HN z83qST+|D=wIrnn~i?H$eCh5k7V3=Vixle7d-?denTc({?rXwHQZ~gnNTXmv26hBJN z`dR983#R1EX=`?MoIBYO>Di}d7Mc@JLgG>9msOM*9@B{LUn|!wE^S_1dM>-vRTlL? zao-kC-!z*UCn(6wHI*zv@yB@6Z8)3<7p)LYuAkm=vo zz2@*~o_!CIr3W{we@9hHWgg_dT)yZ*L6a+w&Uj(8lw4R8Zfn^}9`MMx#*Y4?3g;#+ z27Alk$F5r3VF6G{Kenu&y=5fvc39Z=#CL2;?Jo1qU9yvXN56{u9*UoUge(rNr0$Ya zY?bd`Ms7JX`;uNd%u*#Do?YS%-we%6nK!vRY>5=#adwtNyItpufg&-?5pB~#g&5tQ z3@*kI3l~2{HzLh0*$FG4E#4O9jM=UO>w@#e8@)@z{!Vxcvwo1=<PVKk*SI_N|M<70CaDYawKFy*+e4MJu`kRl*Oo<1b4;$0tg))T zG)1`~@2X!?<%X&nrma&WG=gIyUOBb-QP^EEVZR6s=?ztOg_1ED#~YR%^?QxKaNp16 z+8W7_q^0$ssy8W$qH9h&CRv?#V0R;!Pp%hcea)2*ZX(`07eOcfMBn&0l@LE#mzws8ulE0!@cWc(7On_X77gjMn7mAB5?M| zK-&i@VMsiW`;c(>`~-3*Vx2RJG=qSAbLO90{EQ1$(X!~<0=7}qHt_a!bcBM<`^DUd zRptF3*sh4gJARL~o6wtpXn-DshhRUFVKQ`!Ueu2nX~$|lUOrOK-lMr%ixi z71ssT3Rpq-x`#=u*QF*+VjD4tja|L22*&Fgh3Yp$@?0OUx4 zIq2KHT`=Rl@C&$uA%wB{rf0=QC1i6%6D7?GxOvs)jT1$!a`qT&=H0XP)~q={O^MhN zW7Mj`jS1(o6HGapP}*OwLE$spirlJMnfHi_ji&~sL=_&^2qSU*el~^R z978N1_R=^F>uYL{vc*BUlWKocVJoP)yiCCy-!%1n%BARf#(HT|#zsIk<5%WM| z8Y;ElQ<5=h<+D(QOSWX}#dx5@X zCQ8#f_ph@>`u469U0=%k)=_MAd&b z5Migy25t_&4f{gC&*N?9J3!_2b+41VIX0fYz%_xb;4inW6WzmoZ>*Dd4!taSc>>&g zmOtnf=EymPEx5IM7`GYSif0>dwXE#jS3Re?H~iG8UCMZ=w;#70{d;y!U?m*cOQ`8x z-%IP{KdlpaOhz6b&PjCV;ZRpBZ3q;Ho2x<*{^_SXmN*WLCX+a9JLC63~A%5{q&$iyeOq}Am21&z4(bt8E(u183(eOTm z27e$wGSufL>x90y{GIb}u3+g-CEN=HxXR$h*tI_SeGNe?ZRdTr14EdU>4 zhL-iLihWLpdS?u#SC%h%ot!+>)RN_ac`SVafN?zV$6%R%@G{{jlkFS1PL3NYV|pe~ zMl?q7Lhu3_EVGA~x!xqY^L!EOaLB7m^^8mKybn107+zrz38mHkrXt?G(*ChLz1|u4_ zRRr5=0F-9a*N zSPnb_SS7?XCO6^Aq@MLO*;45vu05pwJBoGJugcGz*BkZt6NdXEmxmZyZjq{nhbWgW z?^S4zWhO=_QnC4k_L@T!X=lgCw~*oRC~&3#e_(j zie`K`%pZRE1CFs^B)Pz&tt9!_%SQ*G3ABSy$dyz` zkSg`GMs_$)IwPBFox;0kOBVZW5VDnBrkXnE z0dk9PPLup#5E4RJ%8Yl#S;bf>%^mYd)d$QpHH~w1L%@6R0tFd1mc)s?hB2BjL0x5-$WoD<1G~4`5YAln4rUl zjk*{e|EYggAxFk|O_Rkv5khR$9-^l!^>#|fWph(7BB#4qO4#PWp^%1G_9eIDQ(SU~ z1}PVLd?Vdp55e3%8U~BQR2e#`jL>%WG}7L8s8QbCIJox>#0H1Yd$>Gzxe9Y)KNS&; zf}uMUord{E4fyR)8%w?nwo!;{<_U9%f{I~SP!AH!`W=nl-tMse*dt@M&)sW2TFJTx z+e?iZZ12AZ+dF-zy>^%Ke{C=iYbY+cok+=~5j z8*OkJe$V$}qx@py;3_N$M>}b;e{DTFjNKUot>K{W=l$SED5Df9>Fg9Xl3sN_+bFvF zd<~6qQ{zn=vq?Llk8bGqYc>kXnyc;ytlR#U!ycd=PJ8AKb}PrK=`7<|_xS$MDAxuF zAU}V{g`df+~kcSl3+CGiPp2>-;&t%@I=K=Ue7KefUHSvZ03x1#Rj} z59rUfvO}D%yMX@-->(|wyqoa)&e~=KZ)@pR&D*V<^FlMiYa-jG(WSPRp$N=8ijsRs>7S}*ukea~ZV+A7 zzLAacNYFi#JjM|Wd*Cz?&R*mYKLFx_FRW2E1jmGu6$+scs%AA{x$1=BHZyykKq)@f z1h>8SlJ_w>4;Mdcn-vH`^-e=J7|`~|vPBp2+pk#eNz^cjN|c7)0|o9qerDlS9Ys~^ ze3u*KD-DBd({vJ~ghGcMZ>;%|&M2=rGr8vKLhZ~MQ`GEPDh3~i_XVb)9`u@76+PmE zIkr0w=H$m=%Zy^>9KYNkx+8p_G{|Re;6DcZD$Pbe<@WK&DeTW{sJu(rwI>ZfMRPd& zh8W^;dfX)`nu009cm7g~^?9zL6KWm#$B=jCwHt^sj zqT?qP!Yg|ObVU0q8)WNXZPWeY=+OR&V%f0A5r=SFH@Pi&b%!=e+qO-FXz?M>r;KOq zbar@SgvMZ=f*8;4u{{V&j)y{5a@65B6a>0$P4Kl4fR0b&DT=8)?ZsYQXO}cNF%i3)X4gH1PDY&fC)1zjYbY3H9z9z~RoxCJ! z@U)sIX~DIK$TEN4_er78lWO_&5vxD&wkyo@EzMJ_CSN} zB0ivBXS4AcDOIk}^}TRi5$sLyAo~_OBQ5nWC&k_4*4^jMap~`4RS`X|MPzIOCh<%2C;Yq&SWxu zfFCIW|J+z&?2R(qA4U9gg!Ir5|6Fqw{PR|o!g9v4r^hyaZz5Yym>m;6AqZ31{E2H@ zN80C!WOD1i#pP0;+!^7*nfZDG9<<*gjjy>9-tBVz_)eC#H&GS-?mIgo4@Eb~(GBkC zhA#JZ_FGzxLY#sEej%;E=1odLXr51_>@lHdh?8z7w%Kd$>G{Fs`oYpbh;8&pGBT@rB@-Y|5Wh}Fw^CS<$@3uPp25UWURxvhsXaooo z><9W;wgk!5KIOmj)lmesRb~3T;tX+@n3_}P26Y3zKiOnY zkh;7Bd|smqsTYTM?@QDnof|P%*ZCYa*5eqH7kSsHEmlC9sFp6y2HS4 zIGqkc+0T(K9)=4sw~ zDmYhMaR)IMZ%}zpvVe=>FuVl5gVKPw#%p*rP(#w>yAVyZ@XnvvCVC*LJ2P!i0p-}l zSAHXB^G@gNL201NQIE9UeXlLR=G(S|>P((ZLBy7nw(cT@M`u_r~* zcI7%0fS-zGBe(2h-_gvY;{ZWIsAT6TR8{U9ZIhF48sdTflimlV9I6Y*3oZ0bh1ivR zZ<+^L=?2TcH542_S3H3ATf%YbfK%lgZj-}r!r8Vc&<5{el1Dz^Tx)NY9A)TM7xzX@ z$<0S-S4{oi5eCPEq=~A5jZP40Pc$5sXWFwxP58aF%b_-<0jy)Tp^N(pOWNL zlKZsO^-3a(rhX)xDfr-WL^nmVWS1+ zBqIz=OXqPx24#e4Ye+_zMq4?_2%#LarqjbQ>wJHf5b&{I(2~S8BS0vO)~W3JHosjS~3AyC4kQfAJCh z*#`D^j^S}&xZPJP$!mfHAivB6Di3R{%is4Jh!~Uw=F-T$n8E{3D!Iy2MpXD=-zo{2~ys?J&z?0uM^9nbN4KrGbzEFcTgqFVrvLb zdr09GHEZXHW=Nu|$5$-L#WyvoB$bE~G~UprC9H_}v5L@H^A|k-9oxhUWuo@Ge6u9^ zDPH>tec!+QvnFIb?@V{C$-*5_cq2)9)-F}w?KA+wBw={;>CukKcy;Nuu? z%tJs2VVGh#?V+=>u1Y-4Hb%z0Q8kYy8G4x;lJ%Ne;g;{=-pUXx>48{CqCQh@+u6S*RV8JaY!_o|r?~ z{Zw*S=>AtrUYr05a_DDwA+Y}G`j$90h2#C$O8JCNS>=n?A^GT5;8kfV{It=(PeUl~ zhUHdit0eYugM0s4`d{a^I1{9~-E(`S$Q@E=t@S4Me89R(Pp$jNO-%ZSM~O*qbK*7= zc!)S&ZMW6#F7WNO%AZ+j5}wd&e&@${V+qu=K&n{y5VI17%DL1ujlE2SfN6ou6vk0( zwocvdJ7|>;1~FhksM48=zjFcouYH|X`Gergyga`r=N=M!#--0vq|Z_YWbCnFToN+Q zQai($m1^KL`e;w=e9k(1q33Gq)L~9Eg(dW$}PvbCvOeoQa^hz@Ca>aU_Bh!@2n5E`|7On z@ybmR`!$@&PN%(MAn%e_ zBgC70=cT#s?)n}_WCK@qzp$;fwZF7G)~kV}humqq2-uSwWIw*Qc+A9OzsCXFHMuoh|DR)(7Ok*7@tB ztdiMzYW;BQ!grh(*AKJW%Odz^B;>&O>;Fa0D^jtEymt@PiypmCu*zT7-z4V^9H2P> z^N{4(g!~^Iat|QGeM+mWv<8m*pqw{KEWUkE&T}g)=d)p;Tbr}V4r9{se1t>DZm*kT ztAp$VYx;pHiya%bWU>()V;3;q?)$!8-hFf3rrm+M9UeowlIc057m#=OzO0wOs2{2u zPKKg}-+~9rM{CA=tm@Q`)>JI^LI1>6CmQBUMhikkB=M-nYT^CdH4P?-5&IoMd;I}e$ zGh-H}*z=JdPY@_P+?qgrqfs9`=Zj+m!yv;3Z}HI;o$Oa6w_HmVRPb`7JNzgIGM^35 z{vrE2C+u+$cE8VCFH6A&!>bb8L6E;ICSXj0(zuxa=bkzuQ|&|{#1}j5c4GW zKvk&X@Nl(bLe-v=>q3?;UHoBYPg?b(TSJ%5S$6ELnTh$Tt|E1>USWSf;&hBNa^G&G zH|SX=z<2y>^U=9`b1KG*1%wy6EqvP2@xk(G&u)B-v%0Sk>E3DmIY;K6hL5?RTe@^v zb;>i6vFl9*I!x@3>TzPd)690Juq-MG2DybwRTbZp5&bZmM9(;`W_l_a9n-}A5!gf4 zm(>SKjSZHv(R=7bjG{^|EZ2J@*|Rclh|maP{XYy~mi^mAe8()NXxMK%;TFbdyV~h@KEX67uDPnkqyfr z-!bivtKvJv;=4V?zr^CP9@neXxDHQ11eegbAlXGq_Wlsr1_veaxkj8i z9}vmtrJ7I;CIV&wRYns*G>$!9^nceP;m+gTl94*8?+brG4#Y9Ih1yMai@oEyD2IZ9 ziTD~-+!!$bC}|KJd6LodrovSXvz;ruIzgcDQdQ-KG}!gY<{6ES?oA9Sgq+zI5s74l(0+~jWg8s<7TH)P+xt%`S~1Drn8=i7zI8bQR4qWwGbY=05v92 zyXY5ERo_Y9H@%u_6k?6j70t#3d(uQ>4!oXqc1?_Zy3(GbOwq6nbV^_00s-xy= zqldWz)CghS`7_#8ngnsNi37oHJ{{`7pPE?hD{7SLbF2I zbH8X7iWg}>TxKdf$CDc8voQn;uzCVZ*Yp;_M%_Fi4R7uY(1LfX0x00u)txVd5jo?a z4D=k+Stdj}l3WuKYOZ*WWw|Cq1n3(=FKHT!SY)t*-xCtMMm|213H0bCy(cNvu>H2~ z3^pwwi2_&?sFT`aCiuZ+@*2w`7)L#PvFgh;>>7=3V<~pNA_8N-TZKv;7x5ABRUWTk zi&x=AYnw-p`)PRI{s%;5mlm{oZ#?LOmP?4s?ucwOZgx&(rwAvX3Ausx^1dGv8m2eRW z?nC`Vs^)J2-AX22Iw#|!Z%4p?*^Dh(%(+z(i&TaAY9U(89S~Kgs&c>Da>+XkQGUbO z*HpP^TuVvp8SMM?2C)|QKwz#Cbm4ImP{1U3?T(ST9PbuT@b6%C0 z`xWqQDhC39(>}Czu#AhBf#$}R+lM%O(?Qkpl%>nmg){tD&7x6=jt9h-6z_tq0tZ7?o?`G+xb54JZw?*A$1MjnP^px>=BIcH3dJ|_R@J~xKOFC&Z!a{e*vB+eI=k11uZ;7(wMKq> z&5(|vWKw|l@u~_Jg?fh-?$glz-@zDhL840`7N>fQcagrSmY#TwQ|S9l8Uv59uF7?` z*ma3br=Lr!EK7>LTIgbn=;y5JvX*msA)uzsqx2XRre#XQvCE|R$)eZb-{@KZD{-Cj=nJ01&hH$QO11U7#71h8dPorC<=q}c3$7Cn~j=O?U6(iNuM%K8}8oOu5q zsl7dKt`6!#CDudmwCYNUK+%Z(NG;KU;u$kS?7!Ig=<2{w|567(f-0@>yzURG9xLc~ zdt;~Nk2wk4CSYqv8c*?KJhnOM@<=*wx;-VzkfE*7qxPl2Rt6d$<$3cqBB?rZ=?ZFh zRr%m^H|1o(6Z_B;@cn~NMx8^5H>eL-6ARplluiyQafx$rNN)G)Ji2G+dBY!Q@(ny& z=?O016Mv6s0Uc)ea+R2)7jwedU4iK@j40j{VauH}KT;A6`2uDWWKtImIglsC;;ZngeR<FDI$YDYY4!PN9t0(v<{j*f zz)S)@aRE&XAs-(+6a+|5wk;6L?iS0&O4@GIk@Ehctsw&w((z+bfBDvsk_XbIX*E{? z?@1Za^_h$>K~q~$ur}j~!`x}9!+(1@tw{h96dBXJl|d-_0dazCZvh}j4A_sr8gj}%5Q0hx(248J!R!|VZZsV1o7sXc|^Z7;jx_HAxH}GB>GV>@D-Vrrdot11UHMiKb zHr+HS!>}O!HBZS}k7>rYL97Bywm0U!d|k*J0iH(Jx6FVHjZYeXyfeM3kp~GJ{9Fp z#lbZ*ussbrgTC8n*_bBVO_Pe{(Y;8H`*uRhj&UB-z3dr6krX$VeZo;}2Z~x>pD6DU zhx86@Z>3Qb3y8C%fS*kZF!z`=sjaDMr@7!Jh>9Zl6VzEkSUy$9T}^ z&{6dd+c;L`ez&&sk0N`QCWdy&@P6aQvkLb+wVgJRy~CfKiDT6q)gqwk^F1lbPX$ro zb`v7!N5l5>6bG0GfW5vFQ7+>XFyL?bBI?p;5FSDy@2%;2MF*?T8-$nphnab~X(;!1 z-vUw2y+LD%1Lc$u_;=~)?rX`Rz}`Eb7sxY51mg73Z9f3RyRB>*$B+yR3BF&6ayG{hupSrJGv5SJ z&I(pQD($YM5HL%oFG;x$AKG*dIRcPI-#AgeI|xZ!g)|vCLgzQGMa02MHkROERk(#V zFz<9B9($7`ZVp^wI4k3f^H#Fa1S4s?bn3IebrgB0+xK1+VwxK_&^DGjqEk_eHoy%6 zsvzXM!$Wb&h(piR2vfOBTLK|&k3QSMZs8=IW8y29d?BJN+|;&`I_ggxXtVr%77Rvk zfo}c-${s4_s7P1NiS(4E9&cq=Xp}ln02$oXnS7ZAr`uV-Tv*eE+fFVe+e>1Pr|ho6 z4ki1J!yf{?-}fbxzZ--<>Gx{+ad?IRr0HI*z0Hd@I2|}5zv9mdw>a#sTdhY&vlD^e zoquD{mw&{>S2gkpHe_$9PSebV7g{v(IPz5iV|KJymL6=&GHQ#r)S!8`V)>rz^VrNQbg z#eTgXqxdi(Zx4|wkLwTZWb#ht-pRUNH8MLrhh5I`505;W{MeX^o}_zX*6Fh{yrrAd z$I9X8E(<$H>)2CPSJQFmo#W1Ss9W!KrtH#j{SCX1 zknYEX98I>h4ZQfXD4y)IF{X>OL99ol|OxRlRr%%cl+B2xO{HgE5Nu_oV-+?KGMSN2;}Hp%yUCbJ-| zc}9PbpFw^rX}TedPJT~HR1IA8CxZQG4&HhpW*(2b-n@-?5@>PwwU8UE`2Um1_aqN; z%|EY12T0}>I#$dq(_Aamc^_N{W=IwP8+(u22av-cH>RuD4Bm!RwBbqLd?wEiPAN>` zfw|}}uCEX3>!c|_`_+^?Sr#ud3T1xp%V+Y;8)e#l&HMPge>w3No@6C!rQs2i)3@q2 z#X0mX_RP(W6e$i84b8`VJa^``0@8x1cy2gs9C;5CL0H56AySI-HE)~S`Kjr5VL(T zOpXgez-vKLN&TB7=BLTo`ME_Nfo77QNlA8E9%9zwMhWhdjD;Od$+8Is+x8-MfkS@+ zbPxLAE)#b1{F@50lzF6cNA^qm<`2b1D(Oz4(UvNOjM3W)r2A<*E{KvQSZsB+ESv@v z)D!pV&t|cIa^kOi~NISaN)OQq+#d!x4-_he}&#VjFvy|aAUf6SbA|fa=YQE z?85f-bf8yPs5RSpA41HyMqyjw0i>PP1}aM2l2wXsV|>@voF3s<9V_gH`u)J=Ii>b zA1g?R&*n~)?K(+yB2?7LyTZ1r3caBvbBGsQ@)K*~l|F?^-|@X>k^jIuF|aq7?o6Y5 z^4{TFvZ_>39C5Kuxve_w!|H^S)e+}PBNkUDROKs|hqO;w5wlKcoKO$9aYfsFLcQZ0 zdzPav0_yF!9!fVzI7J4;pXm&4|G@a1s)auK_MpFwqa9ufYa)n zjv}&;S!7%&=hkhCn}96k2tch&e_WCKxDxT<5k4Rg`!P{TYxyBwSq4h(_bs!?zqj02 z6Q&V?$}AVMg;CH^HBHD~R|hlP3|=G^MGpANEOP1KK%Dnmek5I1)1<{5?miBe0Jwv` zM=bL12H{TodtHn9kg-M@FXWZZf(Ef&XIDiD@wf{CM_6aV)E99++w&v2Hp|&Ra<4r< zhYR8D61F>sD!fz}weW~NOkOhoaQX1eqM6GpF3l^CJfRhy*oEA-XPu^r)X`@yuy{_# z_0{65M}7BNWc^KJIG{<>z?M-ml|Zw6ERus>0O)7%5+a|Y!eSK*=TE=G(=UAEEVAk5 z0Z|RkUj=njbfBB8b(_5GIxVWslPL9XUxG!xeX#lP{GJV|1v8e-;?p$~ngt}UCas;b z0PQSD=$wmT!F;i+p(X zkPgpu`5rRXwcSmSJAP!xIOMy4{Jx&rCc{54E$s(PxR3aoxAEpD-%qRMZ#jGoc>1f4n0=vlKfAnuDP7k&E;D=}5+3 zK$lW9=B8iBZn->`mGE)PUoAIa#De2>6yv1@WI*pkiU8o(i8J0OMD0)oKnviF#t&Qj zkS*)-a*Up)p6~H&`RbR{CvzhUijItUQdM}Lz>ln{xK)01ydCK!?KIN3yrnX92<)EP z_7O+MI}&ZFLd;^RxI%w4%ehZ1Q2&&a%l<%zq@*p331!taLFMUxyJT)J*MS;)itc7z z>XTF$G0XNunC=;mSn?09E+~>F!3H;}FA8EwY0Pm3E}IlFj`ACdN6s z3GYD`&wDC&wVb;eDHm|`ptpoL)1G^lj{tvH`03@L*aMYQRzp$1`O{KWwG_Q&xs+@; z9m4b+L;7lY0{WWEVy5@*0pvV-jtKzu133Qd_bKpP+c*B>u&d+kEP9LhwL%<^<;pHo z?{y}umW`_eM9^{hSHhDWz!O3Ok0OW%I=-)vi{UE!h8G*QS{}hG2S2eB!yH0l6!A9Z z@YO}Qb3bL?$?3>~Mu-YFN(fhPxTKRZVe!o@EZ2EGVuf-`LCFSULyxgfIL7O%MAu=T zVzsPX9XK$cV0r5v_+P(+TZ`jJ{NX$+=dwC40Qh6ykE`ULRt10|{fk<4VTy_=Xt)Z0 z3|;^8A25pOcyWwoP@m1N(C&#A=F%*K&QwDL@(cB*lJNzl(qzmf?4Yv&u4DZYb9`L9x@j=dJqh;c}yMR11!Lxg|g@A)s`x@j{YM$z=F=dtb+WsbCu{0 zlXni`HN8S~;raq^%fP(x2T8{nGC=V=H(g;88h4>=Xz=Ctww6BD)CpebpfhE`X@+$dHyz4P_FnM3Rhn4WR zO?bP}*Sbn>9jc?md5;Dm-xS_z)3|fq>bBYEMD1vAwc$u;$@%ZS)r(37^mq(!p*POu zEMB!JDcjiGr2=$7Jo*c5$z3Y=J!^Zx26oa(d1Y`u zXpZ>@E_`24d@jICOy|5-({x_3_1!50XTJAZ7nPQT;3>58ZLcA)O>>t<1-#^1&02V! znh^XJ*bDRnroMsizK6|>&vEq8ld7hC>LXyUTCOsgKn(hMG$&>jhFmq6iuvOH>-Qk#!2DTC*O(O(ds}h zA*T)!1Up7+s$IPz6!}Z^rnwM(+;^^GUlA&-u(E3WA9XI?xh@<^C0Zk9#X2p*0m~JM zf~G=edO%SdiW)Yhc4jOeyN|s`EAkP2+9_k$74RH)_pYJU3svS}7 z&p6jQqP^M`)17VYFJzi#PO2DQ5mFIVp?0fRRYX@{S+U(jfLXbUsa7r5$O+56x_F|T zy+P!@M8RG*UGZuTG`O$UIwRYrD;$wt4ZB#oLW3@BZgtJ8P#9eb_qkfeXJC^Di*X{g zr5@|VF<<-j?Sj_vjJDz)6y4J5P;-NxQ|%4y9$`AFtk) z50VThWb&^n@s-mSNkoi*M{N)GA&9PG?Jn6v08Ko7M#t% zE>Q_VY=<*NPS%Pohy!{>i zgeNTFhyq&-wYzjeB%57J+0=y_+0;w#zXb{q&~z3a{Y^)ptphXW>R&{dDP*ENvS zTe{&k{s@~-u)WdjT~7aN+uz27-Ok(owwUxEZU1*+;}v!%8t>^tSZ-<)%%TUrWW_|( zjmelaJf`M|weFM{rEo{oW|R*H6Tb+Kfiy6 zV*6?Ln5(MA(w#AE>Q>Fbzx~3ITD`TkNSa?76CrF)t*gpc#2ZUn{-x2PBMOgP4}`M% z++V6sl0uVo6d0}TtbawQ`{x?gP3Q`{6fBcy6oQ%8lUm=FLQ8cYw}--Ft0>iXka$Mz zNjy^%m^0tkr09g0gp>_|Im^DOk#tQl#&QGwCjSp4kpHS?f29Q1aA=Qi4XQa!)N?80 z2TO9C?0AhZ453IpU~O)v_pa1BUZf2uOCYQHToaSIj0(zz229 zg?$A@;0)NrYP33L6FymO?V2~8TAm`Y$F33&8&)TPdglKQ>O%4drtP=cryBR= zg^0qBJ*FZ)?OrJ=u6zj#L08W2qpJ%NF*24fCtZfG2;#+Q0X=%aLE*N}eL;T@Nz*Xyt0ciO5uG;Ci@zHy1dFr~#c zrI9;oX{&|LgvRVL@;yz5{r?&PC%k*Gk=Q!_Rc3W#sf*f_=TlU*O|Pn*{ww)Ia${1= zLrO+#oFw^`5+4!VF-f#V>C?q@A~dqgTi|_fEw0}yDiDhkAq~AsVf@jadtoD+O~x1t z+i*P+{x0%mr{Qvho!i)}X#EEif^!K4*k3j>i9}}k<>(W6Cn){hGaA{{AI%Z>%tkha zJh$gJ@=6>Ylwb1gGooiLUODH73uDUmX?I>;{?|+Qq(*jZ6dJ)z-^eNjSXqtLcQtZK zc)`Ml3x`65DBx1S1JC2SpWRC~UBBEtu91z9fdSP>U3DXSQ+-N8b-ft6sB8K}Fj`AoTs^G;&YfPFOQi);ya3OB4)dU|PITJp*MyBCc z4MSfy^nBgW@A;nnQebeELRidxf|m9TDO*K;a58Dnrwz6qPA0i>zQN|^Wa1$HwsSHX zry`b~JeiPT54^Q+Fd)I(I8OLCRxv$4JY?iA&$G;n@C_L(d;h-m@hS4>1_!1HV}xP zsBpS`?{1&Wn>!40dxA;fcLbXF8irnO=uExtn{QY3Mr|2OQ)Chk6k^L8$Xo!U@vHz+}%bG*o~LC&~l+s<4oD)S?;Ou6sko%7scB z8-mRZq+bv<&HZFR%;AJ2W|1&ZyXWvT|mt;OucS@{o#|7^=v9>w89%y=cdaZbRT2_~HW% zLsn32OV0xh?CR)bda-BmK6VNLBLwZ0c}*2YjfP2F`s~)w<1w9#4(U@{^91!j4hj#E zg6^3OVb!;-XU>@#hE@Qavrr(b`<@1`@>}r7W0wvVsLm?C+Xp@a;$c*NLXQ}q=hVl}Ja&xugbQkSV z-j&fYm|g|cR7t^vhU2sk zxi#4o(we_X?1u}=XMM{5#%;)Hj{B~dJYfCq&UG(or}43&`C1f|O; zjA}^(b{_Mq#O&=uF5Ek?-83n0c#pg%)DZf5qy7gN;Y6fex`W3zqwme}A{39xKVR3Q zvG4@);)nX5-zk^&L%HhD>v?%A4^^5xD(JfZUQZH=vSD%vE;)+_zs!A(RAfIRuXwp% z+l);ceQiHe&nrWN?8yyy&p4t-8L~y3)usNq-bb_Vkq^Fj_P9`h9v}hSP|xt(ckDa$ z9xD}oG}05QcZcc)c|TR4KE8)x5nYttFvy*$d6&)uFsy=RpGtX)amLKE)D+8{n;V?K~-A6h)+ei5|rCJS}yhfqg)Qk zEkij$mjrk@|MzmqEAE|d{D6kiC%LY4AO-IH=(8t{VYdk63a|~r-;?ZS^0y1|;mBWs z9wk({`YC#<*yE<(eA^;UayQj8VldL}8@DS4RoVw7$hmfeU@pS`6%+SRM_Koz?7 zw%c0Yt2(eloczi`$eOT^O<8m#$Ck>zK^E#dSX71wbB2CffAqMY+1_cSJ9|^vhlGiu z&axD1Z5#8OSIbP_-C$Hib0UoYrqbR~tMfGG9VkiP~T7ORp)3_p`of%S#gc z?Y?<_x9>iGkM9wG(6_-q|6J^ME-QL%YZ z@`O6E1Z9OOgi|q6dwR;*&+X|G&N}VscbzS^r~fQ;!=A1W`R(bcrqz@^JxyCI>gm{k zw4Irx{yNA>$WOo?UHEH<*+nd%Dr&N0kgyG5*iof1~I37&aZ7L@L5w zo&=@P%K{-}Y}Y7ZUL=2O+@h(-D+rkKv|yZefz-ES3FD`Idk)k7c;74jG~ZkPT;Kct zxxT;om;1i)mwM9U{576*wV(5($NTM`^aQ`hldkaxJn1@rzb8G>f6|kl4RWBjv3X8y z8>=rI5X%I>xhwo9FCpBd%DTK2}WXr7aYtEu@p@ z-&+SY7yFCNYhGyU^aWHDxHa-7l@`zqrrKw0Mi#1GdHeLu zxOGJ&27J?Vdb{hFX)bD8eti3s%?L2nV)G~ny5?H~7eN_DUT2326yCJoM>-zhllW7u z!;lqv?P996e9kVHPYIlo>u@t4UI~y9AHrbT!__4t3YA zoq$y&d-Ls0k5+9hy|pTNpX>Kqzr1u@zWvoMsl7ye5*D?$i9~e@36Ge&roe7g#pyd% z%{ZW_dJ;Py^__#Rn<`ywAweiUkx!KfDlpQ|E-_->Z$X%~IZYa?sx>d=QpSbx*t8+s zhJ6t(L&UENU3!pjChHTC&l|s;3KP|M%j&}_X!Vf;9AXKYU8malhTQ$lCPvEsk)Y9j zVUzpvrZyW+PIESHzqcP_8A`V>cO(AVZXQaH#(xM#dCH!lC@rGxMIzQts*!u1NaJ)^ zg5QN-AaC7@d!11!m>G{6c5gKM~sTs%G3p6!r=@ zYlo2y8fsj_?;>sA32qK`$!{>aZV0Yc2pu%hN&>7Rl!xwpkGS&V%VQCitL_t|{01!e zH++ZW;sP$j0X|Br&!RBHoBv0cy8joL;WJT~-rD~WX6FA3Oz#I#nC8d+N0`X z2;TqC*kb*e`y+@LPiD-6SjB86;e?|zXIpYWm^s=_Kgg??I=)bmFm(_KDmpD`)R?M* zQnQQv`eBWcjW`$qV9rIi-aITuHe}T9MQBMy$O?t8Fck@mF+b`v@+$~kCQzb%K8$tB zWVY@`T#N9Vk$vM8DP%zs-8wJNuu4)cVHQVMu3;VOuEQvLE78(nmC%-sU`xyhLQBU% zU11p^iu2b$ETcPXC;WP}tp}{C42vS=(r7z??kyusfy;~m_QI5ea(BE%7%ZXlWXLHB zKMU{RZ3skRwX9N=x+G;%`5kcR2M&BGnfrEq&C9nnUnDu=yd|TU8%q9#c^<)>%(3z2 zF+ai^u7r1%e-v*rhPVJ#uV)XEO#vroCbMJwVHnUcqN5Za6eup_zWf)(#ZlBjaq??) z+|E&1?)>Uvc6CW<3B&Fs5FKv+xh0BSdH1tDb-@OxfK!ganYf32m~1fqWp5_2Fn!8x z?Nb$yxMaM0r6ZdWo0=2J+Q8wS?XE&O3}S+5kyFGR6 zuZaxX{%jpWSGG8FmDF5X)LW9>BzbZ`p^De> zr+h`FnWE2BKotBpt@$`~H_PVCxz<}fXa40bDcn4x%B43SfGTi4MGje->}Lz+I9&+u zkW7Nc9C8C2esn(34*i7}}O^|0@uFHIqc}=ybym#Mo*u4>P>!J&U8B`!gBVb?>s{Qp{Nc7uk89x>RzC z9jr!*4M{jD(wWw}PrFr(J(JBQV_Kb4#}EbjSu!sQ((8El3>__-OStAK@(N=gN6MiO zW656`e7^zNyD6eglG`VV9H+HTnM0{9mU(DO`-ODrURQ2GkzaH^HQwijCvd;!SpVWc zO-aIMN($C}&Mn)Pq#)>ZNz9uf=Tl`TQx*NG^4=cbGJ`tOx~=sQw5%d6q2;TOMO(g1 z{xU_)4@Z*cr}d`{1ToGCcU!J2SMY4B)xpBlk!Vwr5Lb?p(ci(w@AvC?MRzamkoGkl za`8b&CL4aN$rfn)V@^|ZDzZy79qLX#gys&B=HN=(1Ad?#_?q0dVP7Zj1`b51>^*Ir zdpQ*~9ws8J4fZ*Li=!R;MzM%1rup{dY4`HBb{Q%|Hp)uP&w;U|OAPmn_cyi=Z5&iE zig$JTUS-F}y~({FcQo`)?zs1}7)pi?ed*-gjre`--$v`M+u> z^T+KFQEYV2`|iK&c?ABi?!doD;NSnhf-ms;|0S(@R&B(o_CLs^68tZIAL&HhWtY3wvoHieg6i!^cRExk;pF~Jam$1wS= zE0kgECkrL4#7>xroN937iUaJIY=bN(gkD zaQVSjL5_Q!yyIN2bEi__$gQ)br`;#Zng@{>>j*dlpTGSSF8e8Y{sEt8zpgy~uYP^> z)AG^2_l{wQe*FHyR@hpIzrjxO_Q&nR`S3m&*w8%KlZlK$>fc}#83$w7z8@eA6Ahp6 z4L)yCZ6M11i#Ep}|0!((Khz{*6oJ|E&g1TP9w&N5^I6PV;k@S^5l(gZ&cgp?b-_al zPP8;XH8)jVoKwEcRaUGkG zFzaEVW4K>Yp-*!i?LRsj9tF^-$!3L^Ge@e;72;p1(p<-oMt-oG_|D%?il@D;x0oyV z)Rd#PSNR;z8I|pAzQA*ToF_Hj_Lt3j+1*4q;g@;tSKI#1bow>>-}j~SCH{%NY5wW* zOR1#kUWa&m`FPLUv;2=DHJ^h0If3hVlu=bUAvdHPan45uu&n!wF$tSLj(9wAQSh6e zVYd?C0L@GqAjdP%=rA3M2f3ItnGN1Y#E?3nH>wHdrI48;XEOzwDxE{?=+Fvukr37? z&9>`9o6}r_C=F(;mB>$N8c**UBTaLKEYE*z!q6(U7DoGw(kIFIf-am$9hP^=PWRYL zRNmQQ?V?*AxyIo>)nlo6%sBPHRk_CWaBMCjAAK{H3r zNrV&fW+pvwCNl6|#B_7m@_cLI31l%gPK{%xSe}pZNtju#y%Hwbb!;nQBtfAFJk7@w z9BzbhF6@u&f5In`S4=FITb_@DOH)SqjM9a1Ooa#e@lHfa7i!99;N5tdT@XZXkC{Xicsl1|S>w7?RHE$6@V*e4Hv zfQi~#iNh^!y``B1CEzsV-WHvBvxJI1bB{W#3RRV(#jeF85J70s2$YOhhr7DHM> z&AS@Q>lcw4F@ajM(Y3VBMa(X-a~H;C$qo7hk9=(Lc<5Z|$Ak3aZu;?nC!ftDgya!b z`*IU}?2lZ=?YNe$+1ZlpJiTY7BzL9Mvr?8ekWFR)Y1@6h*6;=5`G(UH+OO5}N*k@! zQCYgQa?0bP&8Z3QuWQX;keNlw{-f4>F&;v5_-6^YJKUP-hRY@ADMO0X}ohM^4pa3WRk zQLRtrnMAqvPB$j0hd!tceo!0O6LN=YSz4IFEw0z%m{9l8t(X!kvX?eVY^k;!XnmFk zQ#?kIwnFSNQ#DDFg&Fjms%t%O*H#_s%Xm@V_+3>^c2(82Nv|Mshdf*+sZXohyL_mx zHh8!;uvdlaJFqz)mWDkE`ztQQS5o!qu*@mO!F}-e>(B{uYBMOt`DNeNG4WLyTlM-i z@S~N6I)`4a4R!&+s{;Wlwl8%~mdqk`r}F3H$G-Z)ni zaBUyjSsUD0>*i|%AIGr&Nid6!3rCAAjwOy8Y}kqm^9}ctY^9xxp0twMouOhc?s(H=wIOcbC+%y9jQfz))3fa6R7L zBO@}9k`rsTh-Q?pH}%RKVq1c9TH_50NtMOGfU!E$(Qe8hcS^;rq}avLR&C=E1Zkzn zmcb|3X7K5@h5R(c0uaa9$_8l|BH!&5+vwv8A(JS3p%QHZ86g8XXQZ|z&SCeFclN=P z-8bfRzgD3A{02986K+;b&+RZ}-QYZNq^WGuRJ57#47lZXH+z;;BOb_Pkzj7}6Ia7B zW!gmX(_hfh15+r67W2(1{qkV;TN0lPvG+a;sSiud8xo*Uy(x7_&nkykFfB+TjqGwANBYv&EfKSFu#rjE;< zIj2*-`4Xactlv^~0%4Y5u1RxU7JLaYr%dspA*MFS)CS!9+~00sClmY(r{g&~Hsl*p z)COhv(g~j#ZZHDZAZ~6f*b`3AlCybsIb$rZc1|*-UEhl@JR=rH7*f1D2gI?Hgo9ghTz!^?*0t{ z_Y>?e89hdB_;bAJ;`ii+pup#FzwKru^Z=t0tdYFRv?^l$XQWWbZA#?(+ax27se&@B ze2mCtkTFWAj+#w1A07~S-;KxLkDx7$soPvb{MiP<$cTT`U_AK5_mOqj&qfUI#2eGA zuTNXZX(gt&q)6=NNC)rbYhsY%Z;jm~0kU@4SZ%pPn41MtfW1(Kx zN)a_k2sC6jNil07l$oT^&S9r8$rzoCSn6;JHh&Re8#g?!TBTr;L$7ry&7VS3WB4@E z9E$iJ5AEhxyqPy>W}bR?w``SQccLySF78TvL8CaYqvh-Afh?+Z1AcxqmX=|Uw=0u7 zk)IZ}W6mkr@0c}QHQG`1@{*y0XC`QWEvkAa^|O$xe+otN;XLK;{!IKd!+e@aK_>=w{C39pW8ltLlf^3><4d_dK7VXMQq0+hW^q& zRJ~` zBoJ&Y?p?A84$ZD+op(>XKLRC?&rm8%mp?Pwm&sP&yw@* zZ5g?a93QfQ+LZo8M}aSu{I#d*Y|5vU4iO?Deb}!aO56}k+~6Lr3GCa+-c9=ETgJJ5 ziLo=?9BUdL%Vmk`PW_B82|22sOr66+_f&T(&a`n1o|ZzFxYb{=(vIjvcFTSlQT)f ziG)s_?(bB|z2~mFMBU$Sg-_!7)G-%dR`sf3tVj9RlqfeBahRC2c5sonSl9=$z5qoM3Y$w`P zSCFY1+KeuRd3*dFf9Nnz*dU2$K2Q^Uug2X~6Y$)}{%tL2^#eET7@oY9n~NNG;cf99 zbnD74dB->5KaTP}?~!=$y-o2}+7{2v9cx2$e&K9nXx0o6)dJ*l_GU$X4%t5yxxsk+ zMPgwF*BapjZa4sDvVZ3Cpaw$F(%1Cif5dB#N_iPv)}NSe)MXgPJoezQO$8^xJxEi` zUaCOtAF>2HX;~@7eo+BbA;C22w(U|HRF9=tWyCF$*p3=B+L^b)ipL%z@=SxqFC`a} zd{W%g-&qsfS>vujUk|cA($@goo=k)#T2y0kW%kW@uk^8$P0hH2P1PJd!J2EZ?^M;f z;7D&v^*$c&kZ#(Xwb@XIACAvEZn#To8gI01%Wm3~Zk|ts%f|Q%;|rNd=EivU12yIa z@qCW9SEYSI<&)L~5!DoG5xKWF9?HH&VqYrU&v4IVMS7RiTcyz^T?yY_zeUxBwBQ=Q zytyyplQN9s3!KAh-i<)lCLDk`xyA!u`oWWZH|F%cQ?NXk-Ri_%560NBLwiFKl`B?I z?9Yj&4@+bmFZZkv*$n)6+t2w7+X8-uZ3QQfvzZ4aoK%o2B*zt=R56BtJwXWCB{R21 ze5r<3Ax9Ds9RMwrK#M{eM??otFpoV>N;iY9qp}0eyiVxawKJuhSY*k};wDj0NX)o2 z+4!nt?%RJ4&ChPqXnjaudpTU9(YDZrf39DrYE(hnVR6V&m{-iW^uFiin}#)-)=tA$ zz$t7TMs50?)>RIP;hB$LeP6`p+(j z;c=+2!cP*oGo{s6s?*$klQIOjGr0`9AC8eF{0wG-t(sp5_X``#uY@C|TnEUE6>Xux zg%WO|&L=@b88(E_w5>FJ9$)irO2)2U=w%f*qh&kctr`zoXCq^w6~s&c?%hHtyPL) z=T&4Nr~G)(butXPcdkaeoA$mZ3_5PP;I4;57A<@dqwbwUFgh-bdia;Zs6*}uKO!LI zQ+e(22!(_pgbWp)t3pr-{VGB3P6Ui*NUmyhzO*l_=Pfk(;Bs|7Tp=ozMKpg%DL z*SluwG2y0Zko598=9pn6nPWtc>n8e7Ls?|1^dT7}st1&W3}AZ}UEU`&nciaW1RSM1 z!X5AWWXB3ly5I&G2x<(E$M>scKG|3DPo#JZ{~ThCcUeTH1ue1E^K-am(qk;6Bzvii zEiQ1jJKfHn)yr~&niq!{g?2nvuq}ZoW`R~g`UQ0W3yCYIbm2wxplQ4!jqZ_#hK<&Cd~D?fY- zl5eFYV)IJy{x=snRD4daviljtJ5?JNVEPl=_hGRi0s7Plm%&!szFl}w>)pSebJdfb zfXs)vAT%tT6xFm#uU#!+n3L~bW)i!Yns@(x!oQ>H$POf#J^Mh@?j2{9c7@2}`VPhx zOO~{$O{3i`!Yni-VQ2W>ybpY~QDsg9pT*V3nqUgzfRA^|mj{!yd5Ytp?!+#IAu+;@ zgW;E3NxX5BG*czZU6D6CdHHdPPX|}SC(5xZIC+Z2nv9h{11#vQ4lHP)z=9O^3@NuT z#a>3^ybUFdxia4MfL?$@El4NU7B!jge<1u`5{?5NZ~~hi(XO~mZt_FcT|t3cC}*ry zN7piAwJ(G|B&GQbVn{7av8}`cpMh7z;{b9Q=IdO{*U(I1bH|R#xh~^I7f!&&ULDxm zzk#}BFrpO3rUN_iw~?kNs zFT`FM!FLB)bKbtda3+l-7a4bB`ZVLTIAg-E%3WjQ)~>!!3o@YnDxP zH8}lza7vVd*aGg0_)9FRbT7A_aiFira3SQmH|PacU|`pii%G-`O(8^$X?P-E(4lY? z@CrvF*{oLa1(})JKzwO%$&@8fT*}xRR!paSWAQz!3k~0?{mreDcvZ;iZCuX`LUfQlpA)UPO5}10$Piom`ekweEn3j$}mJ-Mjd^TAy)f?mVcq z&yKYlM0XzP=6k5{MYX3ySo*Ufj%8D^zcHL@tdO>yTQt_xtAc%O(O4X(E~-HuRm8Ai zeTsruJBY%$J-Lc=W%s*8Wj?xj7KLsBWAm{Qej{|h5b>Lw*~roxviX)JcfxsFyGrA0 z#>L&B+CtNX-4TA7QSmc$iz}g`WQK1K;RP6H64b(?qGjHQKw>o9`iP^44?3Y(X?@f& z#Gi9YY#OcilH&?A%}%LpT=O3zRGVKPKC1@1@kU0NupEz_mGFqQNW}m^Y$Z}W^W|31c@>!Bgoei8F-lV zCb}PsBL19?qoXlQ%+MaJ*|ei$hG7+YJr?uSm*f3`+6?=^Y}Zswt5h zZjHb}z^+YFTn(mtB^zbPHg)T6>|g&F$*dkoq^Kz*jeOx-)QrqH?fZ9G9-#-kxZJtU zSq)Pk=ZfHC^ILGjX?o(dZ96VNIyn`GwcY#>wa`Zo{2YEG*C5U-I&FL+ z!T2^^_$eYOLQ*4VhfKQgZeq_4zcB*k*y}|vLyFQmZt|JhqMMu|$K{K|^}AycW);Ch zLqjLd%g8~QCR$CocKAD8sJqfZ;NQ0Ly zJ2JJ-%cDXCSeyATBWbR6g_u)d9XBjGiA{V9`_SAWN=Fj7E-g!=u$Jnqfz3G6Ra}4@Ns4@UR=+9>;f$P7DT-jTsn)Oq> z)O`w6j{NlwDzM&j$_3m4s_-{=`T%bxxz=|U#a5raQw-q`xvV%Wj8XU%@UPRBpW(B# zx-$-C+cVRRh|=zjIEF2Wbn_z-<9(2ZHcgiX^3NNm>3mW-+%=2 z(#WP|0~0W}q6=9j7m^s>3gg)VS6$le{fz_m1`6`_!#7 z!eiGXcvp^KYcP06U~a$#t?0^(rwbAIc(|;S4-800XDmqy6?9O;#E%AD5cV0q_?a}YruB4}q^ zQ)jlPIA_2wAUXg*GM2BzKo@;(S@_OMez|OqkkUD(>PCW8>%o|2ipMQ5} z8W#p|YQxflO)UxS7K$heS;6@@t~gKj6cDgy5IGM{26$&2j5VIpAVZEPi}tpQWluui zk!4=3z}oU#Nt`7P zX=zaBlo)eqSsvv}H9JPMlg6~)yFMl6xN_t6cTOoyT`>+Bd^`K5xnRhV_{Gk2!;>R( zUM0rN@T)3_%G#fQV0UdPe2}ceQnRs!9#|F8-gAMBN0^L9_)e=xwdIjgpprI8#)e%ezu6RLn9_cWAp-`rmaGml` ze<-sw!b`dGxPgC*G}g=h7|DH0WIw_f>PPG@W?)(*injR;)k7*@s@FE6O}fTX>w9H` zl1~&V%8j;3&@b;!J)bE$@Q38{ne@AV7>AypnC9YKlnp|RE#3V1Na0)XH=cHkCjukE z|F?uk6n_LBG5p~jj@5j3lw-+OMvS?0FU*ZM#_Nq~baQ+JJ+x*QDhRFS&NMEk2gV4m za+}4NKz3bvYv&TgSl8QErOP5Cl;`C?;Z!v6J-&7CM&Rc@my_&lFd+Q&h25G{_{|7& zDkJlLrZ)rI>hMTPURfS3n?Y!_={RVVb^He!$w4CoyQ=2L`(ACHwmVgbj*Jo_7#9F4${q`!ltzweScpWXV#6qT^nvI|cl z^{(gTPcg}xzmFI^BB)3v{ng@G;Y(e$z*{q-t-sjrUB@VnD^p@NZl85Zd26ae+{9n* zLky)%ME+1X^Bn2RizLn>$28|EyBOivbmrfVbP=bd@UAg{_QXqtk4A`Gl_G0_6Q-dP zFA4B)<%m`_0?X*2m{DRZ0scNRmc{24l8evI6UK7()lBjEbkW5>P9HA1lz#D#Q-@bw zns)Jz#;Xe5d2!suKTc?KUOr)S;e3h6&=O}RNqYn=igX6k2+k;H43V^g|oTJycXEdujxi-|51?j3E6+QK5ZRS~EYw z>>AhyT1C%2hLREZq9T`B*}82BOIe*|+m7eJb0A|rQlGJLHfH-(%Q*PiVhUf zZ8V<2Fv)I33_a`Mj1wkjFr1ZEVLxiTiUn$%rZ&!7V}5xZ=UNp#AP%VEH_xP%DzbUS zni2^FIK4$}(fwP2Y8}EDNy1;pAvIlwnn-uFwij2@(>a$&Z;;^1Is`)grtBYXij{IW`x4@Z$CkmL;b-d5%^}^opl;oR&$)H$*hC9Br(vCZ3r;3?V?&#u?vi;g z#JS_wAv&|vJQ4?Ze*@x%6!j*l={xbzn03K1>jJL!fqzCq^@c~~GF&gfq=vshqMES0 z^KowRI$YO2EcM=ugGxYV{*+YLsg3dza5U!SMqR2g^X^+#enxMGZ#-JZHB?WA$dExF zmNUBz(?rc5fZ=EGUaa*LrFGO{-olWhk+4B0^S|F56e-&Fh!K+x)W!B zdtbyUo=&j^Pgi!~uIV-#9sH=$dn2m#A#EKeZJkvNy;m7LS?L}`Q>E+?(iGETPA4tx zW^~b(NIMvaJMAdEh99-#Zmu-^8B__&)Lr7wXbQtGQ@hzp7$^Q5C+MKzx3Q%%e2fgF zSrRQY6$9@TtKD^#-ViASe7v_Qc2A{yUuBw=Vm*~pV)7N?PvYQPvcEF8zcS!yce^TC zH^I+cUTJtIuC;S$UuEzGe0dgz;vR41Nte>?tvv6FulhvY@1ne40x;l0eqha{+Nj?a z{J!Y@!<-`Zdgi3|-IX{<^N6h5ja0)5_}sKj-eIYTXW~Pr9lDUykr`Ue=P~h3nF!GV zSlYKF?Z27ZmUCw&*R@zI3%^WrU55Y3GUzO2>@s4-GbN48Yz84r&fC6{##2o# zaWOoTbXeK+#*TwIRR_A~?B2l*T(3IhauzTuW{j=+#AG$Y@YMjkZMIT7^CncVWbK`J zzKy-761(1{&%d~Ib*zrjEr0Lk($*cTa<+}jl>M$=bg)_-2Df;Qj^?iFN#BFl0P|dq87TSBY8(MkLVaB@ghOrPL!&1 zQYt4kV;HmfZp?+(^VCD>mBI8%cVcB=kC~N|L2Hk%jLr?r42Qy4K@S+H&_T?L&6pSY ze4e&f5hZVSU3bm9MtHfcdlc0IN8|fFP2JtsmUNmgi(s)>1oMc7#G6LqXA3DYX(xs; z8^f3w8&k)26LS}n=vk(+7ibSl>~7==#r>^ijCnH6980dN+NH;IhPR1l@T4}0y0yuD zbM4W^Z5R6%cFyRMnm-efz(c0|@AMm+Fx0?>b7wSdYg&%c2!9LZ92osyBFh)*F}@*W zx=XG@*sdQcm)r?i1H~Q$Opw2AGp+xc1xSRPw9nhROdO!@$PC z{*{!Sv~%(kWD=gZh7NFUV~Z1!H|(z4#H+`RI8hR2^BN~|CPVWUXa5bw{YKR9lAyNU z(iAq(dW$RDf_0Ozd#7EANEC*SgIGknEq2-H8XEe~wZT8FH6|tm|8s4jnCh3Mw)d>< z3$?X{x@SGq{_5JpQk*l${I$zveS&`TM2)({zjo3#jO7|ENi2da;wgANUv4@=!i~Y8 z0UbgyHPpQ}*uB>6UK`kZnthrK1N$nGw4t4RN)~WuGQ}HhWBO5jinL!ByD#OlH~2|) ztY<5O5Fb)in5E>z*m3{B$mS_h$= zJRK<{rb;Xf$P)I26e7~t6=^Y1Lw=fR^>w=gk-O4 zcUzyc`=nCStF)|}9eS}hF|@WfD>T`W|7N1z)2;7xt?BGGyb^)!M!&6TwD0bWHb)ca zdbT)u1(UeEDAm47jp!VvW4Am-RP!Z-^Xa1Xbxr11BGFC}>y#d61{_(6(guj??W7_l z%78#h#4}fYv^#E6U9>P%>ZVIyK!xFm`SKv-CS6o$vKMI}HBH)uIkCps-49_e)L)z; za1w4%xA+jYNu&ge(H^`-e6V19<}?yPtuwHRN()YhPi+YItBB_Px@fk_*hvr6P@Z_W z&qVFW?YV3FM|s@%Zjrc~O6Xaz_OhPp79~*lpB4v>iB2l@#;LTdNOUqWcHabgV1dXJ zPqSx;bqxNro_I0)5&3PWX@4rb_Mr0IB{8kmBF1mhn%z502gjW#Q?t6Y^*gJg%+;OI z=|>VWQ;XEDOdV3MZS%LC= z_$KX3rke|>HfP%jXLrCL_T}S6;cUtyjk7D`b|uk;GpQ=5#CLbQAmH27_Ccdf%??-W zUIoRfgdIy(nd?(2qNB3^VPTS+-sB76P7$*wBeIR%97CNrp|MOrTs1vASOL!6dFqFEo}+r#i+#)l*O{%RFK+ha zW8Au5>Fe&=VN%^p-U^B&uaLcoxLr5xNIv-+=pe7vTiCr7qe#h8Y&j>ksp}oKf-0K} z!E90yx-`wMk?hK)n<2JqVRw_TJp`%bT-Vc5L@hS)*Rq?pnli5;7x*?%`#;E}?R;h1 zWEu*}E=jlk8tWB0p}K0i`F;C1%Pu=Ddnb}yx@OI_T{~&``sM~RKanx}$>11{h}xg2 zScCh|S^xALy$<(aIR9<_`ViN&<{GCSXB?&rAD~hU?(G-lGWQd}&2GNi$q0q*XKyXf%#M>`2E}~^n92V5)RZVl9*rV9r%TuiJ^&BY)uZs+p%$#;~M{hOOuwQA9Nny(? zc2Cn!b0K?XugE5LFRQ59)?LNBT;Q)qME0T-yIRARRBQtdlcDMm+10u^yc&-K85Aj* z=VBLBocKn4Gm{|g!CE5osxuDBs|tUHxj5$&Sqg(Cf1}!PKxBM_mTipm(<#H%?2L-O z{cR^x<1Tl0;~I8q!s(=wsp^A^PETTDPNv2O7HOEZfknF0GYQ`2QsCV_xnkzlql0ZB zmz_@WGwzIv-Eb3f^(x$X6$joB`4#T8f8h>BT$JKb>o7R%`$Un?zF@BC+y||hDT3GR zpUJz57R(<$HOA4_bj_AJupvTC6+?qk5?4``(Os41T9ez&iBs&U;%x^XKo`tb5tGNw z)AN}3i+kUA-TP34SNDrkTAH6%1jH#TJJg{UIkDc7CtZ`)9hi$v7y(8VXOsMi{wxea zF$Muc5_2dpFC}@!(xtVFaop9xO%gk6(^eu{+K-I1xoWRZx~tSBg7W)?t)3!5i@pX4 zLQ3c9!%s)ZooI@@^F9oF1yR$lh_r{xqeBKMl}MU;^c)*fa@|qqKoW`*xmaI&SbRNK zGl_GbDxbusv})g?Aat_ROF;#KX3tmUGUAtyvL!?1jQk$Nv@XQU^&DneP(x5&WG&keIh~*bx%Z?N#U&#W~O&* zq`kM?gFU!tBd@4Y>P;&}E@zYbcsUvhKPh8F<%q{6;B=&%VLW>-8R}%v?FzVsZ<3g7 zELEI?xb>?Gw$N+l)J_$*JzJpN3K>{bW!qsI&cU7ym7Ml-2~NzcgkTvnbLKXXcgiJ9 zTX+Q6UC>cpo{>MbCs0nxtbYV$ZdE{(eIm}HONMcY7uj+dtDi(=D5X7fj*@j?Xvfm=SrY*Ikt{mq8^>WZ(5XOn>_7PkqdB7sr z0c~naQJ8HmcbAo)hbTE0Ir4C^xfJwVva$7Nc9p*AM6EWe!G z6Ltxku!}}2F3~B4X-qPd9t-p*7o^xf5LGmvUog?O%vmzG*jT|ka=OVm!%>h{q?lU- z;ej|@dEPOeu4Wx2=h~ZlnNK<1Dklz^l0hdTv1qzpP>vYbPn6oWD z-PuLH%(E(AUg#)M59uuV=Q!8R=Kp{RsCj4PxENk5Waf+^T*Q=gVP-Q>HaWuFhd75* z3y}bEN2Fbhacn!NXBKU2F$^lth8#Iq8l$8EUEm~XPX&X80qOffD|BDSoC8lr*z4A& zt**@q+2Ke}EapG7MDd8%VqH{J^C;Z-W%g7$cY(oPl+i9NcgxEmkLNCm2E*&3!@r7D z-?rlLVDjDUxnUI+k9i)vayiJP;YhOO7%zJRHWb(8!X=R@Uzo>3d7?c!%zjM-WLNjK z{ymWcZ(cqg)ALU&nuVBtG`(^Aft0^~HczcpeFxP}jGTQBm|%KJWY5A``&yHuyUFRo zw3#|)pbBIp{KI@5H^Yb_J^H`-Fl8>48NL)5CMnrZ$@{Q>ta$+v^pbxl-$g%<;h435 z%2zn6PeO-6gRS>MT3xJx7J2~pIp1}}X>qu4$;hi~F@j6=N4hh~PPiNi^$&~FTsK?4 zKw=jKtd9j2t0KHIT8uJpR1QF$OWI*(GY@vrOFvPs38@1vn(ZUFkT`qh%dzIGu@f%e z+tPRhC!1|a-weHO4Zd!5AFu{oAF#h8b5>~QEIst9HTWvtUT}9?+0Ep`yQ8;{=0!%FJC1{IF^LhOEf*)B!xTmlt(por- zP5>r+bj;yc9J0YQ0Y6(P{j@dswAI~;(oO7QQrd7XmVLqsw%Gmzz>?To;d8NYS~XuG z9!GcjyVV-pYIXkysx4(_3ylzrqPi9m14W%iVxTbX4c0c`t1x^MRXIiL9~Kke{~Bws z*6Lml5HGV+2#D4Mc;H9(2e3!2Sj4&18F?zU!V0&0A056%cs%faMGtjOco#^AGuA(4 z!k=E8^6-toU1JTdu?F_uaIdklS^>59V{1-|O4G_sS_DsyMK5qC);F};igfA$uN2NU z!Xd0*Ck^N%2RhE3O&q$<8vJ*B8zcKV@)S}n7sA0M{=>HC`s~X;w5L^ULjn<{y|lt! zT5fkyPDS{=SWa#~Oml-Gt}u2D9JEeFT4zEncY!j5VxdU}@f`}3Hhq3;Xtp(&573E) zAC0aHf^$B3>7I^RvX}jQ4QRo+D!EgSamxqU=~k}PB>cbXUvYy;oos6`+v?7+2KKbE zXN5W^iL#Gby=zb<{~OYGV!XW!1A^MqL|PlQ(kdl%`%W?bkuTtwkFg|i?M)idklq^9 zTiqIrM3_A)KuQxKaW=gzaiJ3@U0qs1ihaJuCfQzMDW3sHRSg_fC-Mtz)8X*S&5t`Q zBQns12yPF;^T=>2mKQV0HYlv&^9W{W_%_1erb5>Atch9JEpd2^&Tqu8eJd7TzqB<- zTixH51@`=g?If+Ulr@W7*2H*jp|Q4Zmj!Q^1@`{V{bd>JAq7IO4BadX4&z-DIZp-> zB+#`mUwBXBj#S4?_BuJybMYT``WSwnZw%Cyf2N=kC0BQ$*IoW0CJ-5;u?TnfkgW9q zd=Fto&X%r{nx2GmJeJESM@Dxpqat)_=v-OwuW08*nJ|duirlga?jupD;-%JmmKVLm zXbs)5TJ0_OsWRNEdT~Ts@x7rdw&o+uToJeMCG$(M@L;1#%gRv=OexxL3<5AfR9hji z-jCmfFaP)*`%W3eq#8(n?_xKgfS4&sG<2XW*jEAZyZ2M>^JW{rR)_Scz)`s#NLDh&l^u=^`;+q{%5E1d#4?E zUe~0rl}RX$OhP-zEeIvs5)nI^SZ{QA(+q}-F(tTF)f)B+xNR*9wwJjdFAKczXLd5d z4Sb_@B|EYvQz6_Wje-+k)#<*5h(vUE;MuAzw82GY&xW!fSLQYYz$G?W=v*3ECX+pE zHhID!GE{*>|18Be=LPl>z-rIm22i`gzR#%+$SJ#`ZP#-OyBt`zZ%cd!G8I`G(Kf

AOHUP{o*fRv~Cd6SN4qBLKczQ~18jMH7B*O| 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�^0xBfI3Fv>?00`5tR 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 z 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 +#include +#include +#include +#include +#include +#ifdef DISPLAY_ENABLED +#define SCREEN_WIDTH 128 +#define SCREEN_HEIGHT 64 +#define OLED_RESET -1 +#include +#include +Adafruit_SSD1306 display(SCREEN_WIDTH, SCREEN_HEIGHT, &Wire, OLED_RESET); +#include +Adafruit_NeoPixel rgb_led_1 = Adafruit_NeoPixel(1, 1, NEO_GRB + NEO_KHZ800); + +#endif +#include "esp_partition.h" +#include "esp_ota_ops.h" +#include "esp_system.h" + +String ssid; +uint8_t mac[6]; + +// Create an instance of the server +WebServer server(80); +bool displayEnabled; + +const int BUTTON_PIN = 0; // GPIO for the button +volatile unsigned long lastPressTime = 0; // Time of last button press +volatile bool doublePressDetected = false; // Flag for double press +const unsigned long doublePressInterval = 500; // Max. time (in ms) between two presses for double press +volatile int pressCount = 0; // Counts the button presses + +const unsigned char epd_bitmap_wifi[] PROGMEM = { + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x1f, 0x80, 0x00, 0x00, 0x00, 0x01, 0xff, 0xf0, 0x00, 0x00, 0x00, 0x07, 0xff, 0xfc, 0x00, 0x00, 0x00, 0x1f, 0xe0, 0xff, 0x00, 0x00, + 0x00, 0x3f, 0x00, 0x0f, 0x80, 0x00, 0x00, 0x7c, 0x00, 0x03, 0xe0, 0x00, 0x00, 0xf0, 0x00, 0x01, 0xf0, 0x00, 0x01, 0xe0, 0x00, 0x00, 0x78, 0x00, + 0x03, 0xc0, 0x00, 0x00, 0x38, 0x00, 0x07, 0x80, 0x00, 0x00, 0x1c, 0x00, 0x0f, 0x00, 0x06, 0x00, 0x0e, 0x00, 0x0e, 0x00, 0x7f, 0xe0, 0x0e, 0x00, + 0x0c, 0x01, 0xff, 0xf0, 0x06, 0x00, 0x00, 0x07, 0xff, 0xfc, 0x02, 0x00, 0x00, 0x0f, 0x80, 0x3e, 0x00, 0x00, 0x00, 0x1f, 0x00, 0x0f, 0x00, 0x00, + 0x00, 0x1c, 0x00, 0x07, 0x80, 0x00, 0x00, 0x38, 0x00, 0x03, 0xc0, 0x00, 0x00, 0x70, 0x00, 0x01, 0xc0, 0x00, 0x00, 0x70, 0x00, 0x00, 0xc0, 0x00, + 0x00, 0x20, 0x00, 0x00, 0x40, 0x00, 0x00, 0x00, 0x1f, 0x80, 0x00, 0x00, 0x00, 0x00, 0x7f, 0xc0, 0x00, 0x00, 0x00, 0x00, 0xff, 0xe0, 0x00, 0x00, + 0x00, 0x01, 0xe0, 0xf0, 0x00, 0x00, 0x00, 0x01, 0xc0, 0x78, 0x00, 0x00, 0x00, 0x03, 0x80, 0x38, 0x00, 0x00, 0x00, 0x01, 0x00, 0x10, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x06, 0x00, 0x00, 0x00, 0x00, 0x00, 0x06, 0x00, 0x00, 0x00, 0x00, 0x00, 0x04, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 +}; + +// 'checkmark', 44x44px +const unsigned char epd_bitmap_checkmark[] PROGMEM = { + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x01, 0x80, 0x00, 0x00, 0x00, 0x00, 0x03, 0x80, 0x00, 0x00, 0x00, 0x00, 0x07, 0x80, 0x00, 0x00, 0x00, 0x00, 0x0f, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x1e, 0x00, 0x00, 0x00, 0x00, 0x00, 0x3c, 0x00, 0x00, 0x00, 0x00, 0x00, 0x78, 0x00, 0x00, 0x00, 0x0e, 0x00, 0xf0, 0x00, 0x00, + 0x00, 0x0f, 0x01, 0xe0, 0x00, 0x00, 0x00, 0x0f, 0x83, 0xc0, 0x00, 0x00, 0x00, 0x07, 0xc7, 0x80, 0x00, 0x00, 0x00, 0x03, 0xef, 0x00, 0x00, 0x00, + 0x00, 0x01, 0xfe, 0x00, 0x00, 0x00, 0x00, 0x00, 0xfc, 0x00, 0x00, 0x00, 0x00, 0x00, 0x78, 0x00, 0x00, 0x00, 0x00, 0x00, 0x10, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 +}; + +void IRAM_ATTR handleButtonPress() { + unsigned long currentTime = millis(); // Get current time + + // Debounce: If the current press is too close to the last one, ignore it + if (currentTime - lastPressTime > 50) { + pressCount++; // Count the button press + + // Check if this is the second press within the double-press interval + if (pressCount == 2 && (currentTime - lastPressTime <= doublePressInterval)) { + doublePressDetected = true; // Double press detected + pressCount = 0; // Reset counter + } + + lastPressTime = currentTime; // Update the time of the last press + } +} + +// Function to switch the boot partition to OTA1 +void setBootPartitionToOTA0() { + const esp_partition_t *ota0_partition = esp_partition_find_first(ESP_PARTITION_TYPE_APP, ESP_PARTITION_SUBTYPE_APP_OTA_0, NULL); + + if (ota0_partition) { + // Set OTA1 as new boot partition + esp_ota_set_boot_partition(ota0_partition); + Serial.println("Boot partition changed to OTA0. Restarting..."); + + // Restart to boot from the new partition + esp_restart(); + } else { + Serial.println("OTA1 partition not found!"); + } +} + +void setupDisplay() { + displayEnabled = display.begin(SSD1306_SWITCHCAPVCC, 0x3D); + if (displayEnabled) { + display.display(); + delay(100); + display.clearDisplay(); + } +} + +void displayStatusBar(int progress) { + display.clearDisplay(); + display.setCursor(24, 8); + display.println("Sketch wird"); + display.setCursor(22, 22); + display.println("hochgeladen!"); + + display.fillRect(0, SCREEN_HEIGHT - 24, SCREEN_WIDTH - 4, 8, BLACK); // Clear status bar area + display.drawRect(0, SCREEN_HEIGHT - 24, SCREEN_WIDTH - 4, 8, WHITE); // Draw border + int filledWidth = (progress * SCREEN_WIDTH - 4) / 100; // Calculate progress width + display.fillRect(1, SCREEN_HEIGHT - 23, filledWidth - 4, 6, WHITE); // Fill progress bar + + display.setCursor((SCREEN_WIDTH / 2) - 12, SCREEN_HEIGHT - 10); + display.setTextSize(1); + display.setTextColor(WHITE, BLACK); + display.print(progress); + display.println(" %"); + display.display(); +} + +void displayWelcomeScreen() { + display.clearDisplay(); + + // Draw WiFi symbol + display.drawBitmap(0, 12, epd_bitmap_wifi, 44, 44, WHITE); + + // Display SSID text + display.setCursor(40, 13); + display.setTextSize(1); + display.setTextColor(WHITE, BLACK); + display.println("Verbinde dich"); // "Connect" + display.setCursor(60, 27); + display.println("mit:"); // "with" + + // Display SSID + display.setCursor(40, 43); + display.setTextSize(1); // Larger text for SSID + display.print(ssid); + + display.display(); +} + +void displaySuccessScreen() { + display.clearDisplay(); + + // Draw WiFi symbol + display.drawBitmap(0, 12, epd_bitmap_checkmark, 44, 44, WHITE); + + // Display SSID text + display.setCursor(48, 22); + display.setTextSize(1); + display.setTextColor(WHITE, BLACK); + display.println("Erfolgreich"); // "Successfully" + display.setCursor(48, 36); + display.println("hochgeladen!"); // "uploaded!" + + display.display(); +} + +void wipeDisplay() { + display.clearDisplay(); + display.println(""); + display.display(); +} + +void setupWiFi() { + WiFi.macAddress(mac); + char macLastFour[5]; + snprintf(macLastFour, sizeof(macLastFour), "%02X%02X", mac[4], mac[5]); + ssid = "senseBox:" + String(macLastFour); + + // Define the IP address, gateway, and subnet mask + IPAddress local_IP(192, 168, 1, 1); // The new IP address + IPAddress gateway(192, 168, 1, 1); // Gateway address (can be the same as the AP's IP) + IPAddress subnet(255, 255, 255, 0); // Subnet mask + + // Set the IP address, gateway, and subnet mask of the access point + WiFi.softAPConfig(local_IP, gateway, subnet); + + // Start the access point + WiFi.softAP(ssid.c_str()); +} + +void setupOTA() { + // Handle updating process + server.on( + "/sketch", HTTP_POST, + []() { + server.sendHeader("Connection", "close"); + server.send(200, "text/plain", (Update.hasError()) ? "FAIL" : "OK"); + ESP.restart(); + }, + []() { + HTTPUpload &upload = server.upload(); + + if (upload.status == UPLOAD_FILE_START) { + Serial.setDebugOutput(true); + size_t fsize = UPDATE_SIZE_UNKNOWN; + if (server.clientContentLength() > 0) { + fsize = server.clientContentLength(); + } + Serial.printf("Receiving Update: %s, Size: %d\n", upload.filename.c_str(), fsize); + + Serial.printf("Update: %s\n", upload.filename.c_str()); + if (!Update.begin(fsize)) { //start with max available size + Update.printError(Serial); + } + } else if (upload.status == UPLOAD_FILE_WRITE) { + /* flashing firmware to ESP*/ + if (Update.write(upload.buf, upload.currentSize) != upload.currentSize) { + Update.printError(Serial); + } else { + int progress = (Update.progress() * 100) / Update.size(); + displayStatusBar(progress); // Update progress on status bar + } + } else if (upload.status == UPLOAD_FILE_END) { + if (Update.end(true)) { //true to set the size to the current progress + displaySuccessScreen(); + delay(3000); + wipeDisplay(); + } else { + Update.printError(Serial); + } + Serial.setDebugOutput(false); + } + yield(); + } + ); +} + +void setup() { + // Start Serial communication + Serial.begin(115200); + rgb_led_1.begin(); + rgb_led_1.setBrightness(30); + rgb_led_1.setPixelColor(0, rgb_led_1.Color(51, 51, 255)); + rgb_led_1.show(); + + // Configure button pin as input + pinMode(BUTTON_PIN, INPUT_PULLUP); + + // Interrupt for the button + attachInterrupt(digitalPinToInterrupt(BUTTON_PIN), handleButtonPress, FALLING); + +#ifdef DISPLAY_ENABLED + setupDisplay(); +#endif + setupWiFi(); + // Set the ESP32 as an access point + setupOTA(); + server.begin(); +} + +void loop() { + // Handle client requests + server.handleClient(); + +#ifdef DISPLAY_ENABLED + displayWelcomeScreen(); +#endif + + if (doublePressDetected) { + Serial.println("Doppeldruck erkannt!"); // "Double press detected!" + setBootPartitionToOTA0(); +#ifdef DISPLAY_ENABLED + display.setCursor(0, 0); + display.setTextSize(1); + display.setTextColor(WHITE, BLACK); + display.println(""); + display.display(); + delay(50); +#endif + // Restart to boot from the new partition + esp_restart(); + } +} diff --git a/variants/sensebox_eye/variant.cpp b/variants/sensebox_eye/variant.cpp index 42269d11357..c802c614cc1 100644 --- a/variants/sensebox_eye/variant.cpp +++ b/variants/sensebox_eye/variant.cpp @@ -26,9 +26,25 @@ #include "pins_arduino.h" #include "driver/rmt_tx.h" #include "esp_log.h" +#include "esp_partition.h" +#include "esp_system.h" +#include "esp_ota_ops.h" extern "C" { +void blinkLED(uint8_t color[3], rmt_channel_handle_t led_chan, rmt_encoder_handle_t ws2812_encoder, rmt_transmit_config_t tx_config) { + ESP_ERROR_CHECK(rmt_transmit(led_chan, ws2812_encoder, color, sizeof(color), &tx_config)); + rmt_tx_wait_all_done(led_chan, portMAX_DELAY); + + // Wait a moment + delay(50); + + // Turn LED off + uint8_t pixel_off[3] = { 0x00, 0x00, 0x00 }; + ESP_ERROR_CHECK(rmt_transmit(led_chan, ws2812_encoder, pixel_off, sizeof(pixel_off), &tx_config)); + rmt_tx_wait_all_done(led_chan, portMAX_DELAY); +} + void initVariant(void) { rmt_channel_handle_t led_chan = NULL; rmt_tx_channel_config_t tx_chan_config = {}; @@ -52,20 +68,49 @@ void initVariant(void) { ESP_ERROR_CHECK(rmt_enable(led_chan)); - // WS2812 GRB data (green pixel) - uint8_t pixel[3] = { 0x10, 0x00, 0x00 }; // green rmt_transmit_config_t tx_config = { .loop_count = 0 }; - ESP_ERROR_CHECK(rmt_transmit(led_chan, ws2812_encoder, pixel, sizeof(pixel), &tx_config)); - rmt_tx_wait_all_done(led_chan, portMAX_DELAY); - // Wait a moment - delay(50); + uint8_t pixel[3] = { 0x10, 0x00, 0x00 }; // green + blinkLED(pixel, led_chan, ws2812_encoder, tx_config); - // Turn LED off - uint8_t pixel_off[3] = { 0x00, 0x00, 0x00 }; - ESP_ERROR_CHECK(rmt_transmit(led_chan, ws2812_encoder, pixel_off, sizeof(pixel_off), &tx_config)); - rmt_tx_wait_all_done(led_chan, portMAX_DELAY); + // define button pin + pinMode(0, INPUT_PULLUP); + + // keep button pressed + unsigned long pressStartTime = 0; + bool buttonPressed = false; + + // Wait 3.5 seconds for the button to be pressed + unsigned long startTime = millis(); + + // Check if button is pressed + while (millis() - startTime < 3500) { + if (digitalRead(0) == LOW) { + if (!buttonPressed) { + // The button was pressed + buttonPressed = true; + } + } else if (buttonPressed) { + // When the button is pressed and then released, boot into the OTA1 partition + const esp_partition_t *ota1_partition = esp_partition_find_first(ESP_PARTITION_TYPE_APP, ESP_PARTITION_SUBTYPE_APP_OTA_1, NULL); + + if (ota1_partition) { + esp_err_t err = esp_ota_set_boot_partition(ota1_partition); + if (err == ESP_OK) { + uint8_t pixel[3] = { 0x00, 0x00, 0x10 }; // blue + blinkLED(pixel, led_chan, ws2812_encoder, tx_config); + esp_restart(); // restart, to boot OTA1 partition + } else { + uint8_t pixel[3] = { 0x00, 0x10, 0x00 }; // red + blinkLED(pixel, led_chan, ws2812_encoder, tx_config); + ESP_LOGE("OTA", "Error setting OTA1 partition: %s", esp_err_to_name(err)); + } + } + // Abort after releasing the button + break; + } + } } } From f330585717ee9758b15a0ae8fe939967937a4539 Mon Sep 17 00:00:00 2001 From: Paula Scharf Date: Tue, 15 Jul 2025 11:14:26 +0200 Subject: [PATCH 125/173] feat(board): update PID --- variants/sensebox_eye/pins_arduino.h | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/variants/sensebox_eye/pins_arduino.h b/variants/sensebox_eye/pins_arduino.h index 583cb38283e..99984dbc6a4 100644 --- a/variants/sensebox_eye/pins_arduino.h +++ b/variants/sensebox_eye/pins_arduino.h @@ -4,7 +4,7 @@ #include #define USB_VID 0x303A -#define USB_PID 0x81B8 +#define USB_PID 0x82D1 #define USB_MANUFACTURER "senseBox" #define USB_PRODUCT "Eye ESP32S3" #define USB_SERIAL "" // Empty string for MAC address From cdf606af5f4424a2ea568734be290e00d906c538 Mon Sep 17 00:00:00 2001 From: Paula Scharf Date: Tue, 15 Jul 2025 16:45:03 +0200 Subject: [PATCH 126/173] feat(board): updated tinyuf2 bins --- variants/sensebox_eye/bootloader-tinyuf2.bin | Bin 22352 -> 23120 bytes variants/sensebox_eye/tinyuf2.bin | Bin 186912 -> 193056 bytes 2 files changed, 0 insertions(+), 0 deletions(-) diff --git a/variants/sensebox_eye/bootloader-tinyuf2.bin b/variants/sensebox_eye/bootloader-tinyuf2.bin index c7676c718f2383a78785da38dcffde52ca67eab0..7a83be05436d3ea295c2a06e4ffd5a814c145f42 100644 GIT binary patch literal 23120 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 C4c&VH diff --git a/variants/sensebox_eye/tinyuf2.bin b/variants/sensebox_eye/tinyuf2.bin index 6b0fb1bcd0417ce9175fc75bf09b49f7516ab078..67c00aa39f0644371b2aeac5b2cbd898970a4481 100644 GIT binary patch literal 193056 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�toB(t4TH44ReY**I&WYHW7 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 literal 186912 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 d9 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νoK>^ZCdccnlY$kw|K0708fqyvzeRgwf{I- zuVe&HGk7?s7+_}Q$GC-cZ75QOIn_a>ZD*tj@n=5&>UF||v`4{3haXVpBw{?I?%SzX zVLYcR%^PNa0l}8i#Ul$w6pToE4)6|CUTz7<<)HFH0*onrZWQM-k0={%O3<3(7!sw# zq>|xCXUdV$@ym}_+?W&zmHc*28p~0_ngTRoop|hvfqkvF^mFk=+5j$Vo94!FzDN7{ zZS?b2^lvK+UULqN>&CoMLIVL_(7)a`0KJemhpgc&%__}(-6aRsZ%=P&lpnq-G$bH4 zmT&U_mhY{7*Kmr)x5`5k!(xR7yzcPJaq-xy0f6h9nHHZUk<6P38Yr~52HYgvv{m*0I#Fn+9$=|(m0t9TrX3A z#jY^B)K63iWeLSq^V$P{}~i}sI4yPw3_(I~Yan<37aVEZXq!8sfQgn0#sBzA?3vv}v4 zW>oxF@mT5r_>X)BUU%r26;wWv-5XR|kUph*w($XOVy(NrsT+BN&2~ogC4Qb^P_NXZ&{Ph64%oqBUbl}+>gc(8s zUBaJZvFa=o_BQ-YuTr6#X#{8j@3ggY!wmsRb1G&|`cpb{COl2o*{uJ+byobFNY+`r zZ}5=XFi&3DN0hT+eSJ*V7xGH`8`^?ekcPj4)2xm(W;@w9X=54|vfgq&uyCJeu~x8V zjoEacSjZMA!V{&9wD{lp@C`w(4b93rb?KE+9%WR?qk5NZ+JVc!cB5Sd$^c<#rawU@ zQLyjQ2m1B}fm5Ob9#t5H7JzjXMry0luz~OoNdoG4NC??Yk2P4+Ixh3^cSj3mIWM@| zg^-lE#!xEqk&vP%^XMG#-TNgv3s|RcwkJYZgjcP&?s!_ZL`=u>#qso#I&|f$MC7R) z6dz1z#7TJjxYXneA|VY!?L5hn^tXP(c`gECSOUAtE*Q~4Bp`MVrRO*%JU1w(zH+m} zbe@X_=e_eFWE}&tR36(dQUUjIB#PK9@ejKzF26KP59YCCuYvUfANGKcKDCb9zcex~ zwUO&w8W|JQ$FWNzN1%@wt`+b$#^pA&qiE1YIX@}?$s2j4HbUL}JLaIlhmm40X8d2m zMCjuYAAot4whfETWOYjC`pa3$jdAlqAu)a05C4sEeOqJIFbgj>?W{ zkmxDdpi3h+!myR0q6P|kWmT{(~qXeh-c@8`}B+QN88x=x`L@}-Y06A~r zaWbU8<9vZJ1rVK}CaG*cnPXyHT;9@A_y1(patjhayeG!M6#+D-+BoZW$S->fNkN!5 z^GK!su!pgH_|pd zL(`&81i$hEzgj{>R-LRfF(PcEAnZM=RGn!G`l z>B|adEyKmNG2QJ>_nL|#OkSn&?m1|j3+XThO1pKa z;}hb$9r^}pSCg8rA~i?VxaYfOaG!JU2kPD+a^I7>zb@51P;nj=M~!(@994}(jYo`q zA5~>$%|uNx#0=NKJ@>!snzmHTKM|C+L{yDp*8M|-v$ZJZU3tw5UUx4TDO>8jo}vh@ zfEzczYO&L*bSxE7BPWV4b6Qm@yu>-}H8JfK5f#ewF?o5csVD|Lz$IXFGJiOUqspsp}wfBhikVb+(6uZS3;gXo9u_utZ~bPQoamjrOj>KZROFrW;Kn;Le-_uQ zPe_Z0nzk65{H5;b>`8KSs~emc>7B;HiXtl>Sd7T;Q!R;S7^^Ux=(=>ABpE=26N)=? z{>-HPAg)Pg@kE(Pz)K+}xn@##=J%qtXrv>W#PhR5@`PBc5Q9r)`hJJo0*!n{5mt@n z!H?C@;4-dLHHBfuX0SPvkj3zE`*R+|nnO=lr?>4lrs+-A80gR3Me_O!5mjU*y)NfA z9oddzlWp;GY(CilTKikFtW65gh`20v&PCUCd!u$ z&^x$z2@YXSrPl6NbWO5Y!t%p3Rwe2z+om9L*N$dVY^yTZ>#Oi?X?Mx$LOxk#4nK{G zj-0ki9DBFYm~J-}jcn^g6}=>G+DftX<0sa9Jj?Xix2VI$I5t09K@`?3#?5ipXC_w8 zt;V_YsZ_EbISR86qAffVsNeqVw46n4PGbKSVOxe3&ZLyqGv8_4P<@z^C1 zuM^xnYBda_xGLjel^Y*bxY$qno^~zhKh+m{%4%S-V+pgFix)GF9#?ikQrQn=k0LPK zaZp8(-w)!kAH>^^B@G&ypFVYqQ?2#`nFxj2kIi1!oTZzzU@-#ZV3uxzGy#x@C9nr% z2XJ=af_*Oz5E?pnw!VPiD5FOCp2Z7a1Z8cA9p!rt_93wM#EkO&6?TF=C&2zZ?6gkN zqkQuQxI0MnDBKJ(Od~t=2tv8^A1SMaP#a-!NcI+J=*d7 zr|9Ivvno$9BQ195O_-^BG}Mr92YXX~4mhc!$^M#}J1C9b-%awznsiJj`Pv0mxKeBtuBD%|BJY|udvZ(!K z(VaG}U`rHiiv^p*Qy$S>9^)yG^pr>KFOTk=tQ96F3X>PNPIkmzOp)q!I8WaO{fi;` zCG&wvde{T*<OFN^V)Mf%IYGA+Te6=J4mK8S~qZD?{Sq_@Bb`&M*% zjCqQ%blJ%Ou3LaRgh?P3O!pj#{7=p0vopb7+BEtzOSZO@h20__JwUWGteG`8=)UDFv zdVkOLg#`~ja+iIHcVWbXj|}7HfjQaVIok1kCnnm;a=e6%!M{19zJ=Zo72c%o^tx~$ zXC2WUi00S=?Q{zZm@o@d?{mM%zf?f8%d;lH!6{#guv+a!R!5#Sk;+XX_qYCyWYUTK z+#nsqH~-w!8rU4bN3 zc%LRMe*WW=iA>cj{J~b^XmI>1Z;qwS#&TvoG zguFSSMbizHml7Y*3W~e(os2L-3zh7`^yJL1#cgtfB6Gj!k7uyNGW-A947c=NEE<(T z?sI1|&z774WOfB5hm6w~M3fw0msH?j&}WR-nY;_fhCT)YZJKl!<~8j=vk%ES=s|M-1sJFgq`h%kpwk^W^M;oG~W3 zJTK=?Q_fwg3@pkz{pI8_f!HA=wjrqX(^OP>C%bt=d)NBTt^>xdcK`NNM=%K`I#I`H z3ZPgII{df3oZ5x?Cno_&(WpE%!uinw?+5FRJKK>-Q`hEy`#Vz|*R~CYlc1({LMI>v z;QWmO9KYTgT#(QtaGJW1wh`~V^{-nZoe4xTC|9mB1fAO07|IPhdUXB%?Z!PSvRZZ% zzWIc2MT2Z^Sh8Z9f;;3`{2_`hi3(41{}z-lVSoc=ZwE`r#LSEaQJv6aM8!wIWs zYHHy2$JY&dD`@4%Put{%Ff{Qcvafn`d&6f2ndHWj;R*x$wqG^v>AzwM7?h;#m`1at z<{|Gfn(*CNN75JX%w{ zTQsAKq~}iqdtcgrPN!y@zaFb7F@$DL{cBmMLBUm{WYV&Y_hF-4FCaP8%gj;~;MyuL zVHVJ;Cp29tIHfat*lj9D9Vw9$lI3}GWTfM~LmcCzjsmr&SsCahMRTCP$XdtV+jwnH1teK0KM5>6=(b$XU{vYF!zCVk; zhZb&*9jA7@JupcchuJ{8Ob!L%`IKr<8|#$IrdBPj`ioN@QDQ6g$RkrDmexPHb3vWZ z{#nERRiew}t}EWLO}*bM%RLg(zNdY^-@docx!2|I{;`XT{T*s>hiF|#>szYd#Etbh zxOVTVKEg%3gUf5Rz%tjhxJ+)9N1fLs%MKlUe-+x@^DB0gWfGk!E%fxWzGtAuUNJ@q z+C*b$nGn7ky!wz*BrkgsBq%z&bcAAzDnE39?2qLfdmc(dxM*Pxp%WCBbvMQ(MRA$# zXKZ*IH5E`WIgX6`zd(Cgje$6ea(=>E##!t?dtm<%<~fA@{O{%;aEF-be1`YCO#t~# zA*Q%T0kMHirJX;uHtz6vaOPgPDZ)CY-z#DmqIqbSGUP z{Jw*{EeTI{+Z4RJ*BXDp6*nc@QrKIc$tf4fESB}#4UDh&nL16g%9EV$oUO#Lx~AcY zr*8pY0-P}kk9BX4!`8P)g>c6m|3NRfrmYKo7I36ay*5JP?1Q+1B!oaXr0dj`SBb2^ zO`k@|+v{BqHb6)s+!9o$_H7fRXF}isDeI+pzFC(B-Bn48)L*#jGT&)nw;vE8h?Qj1 zzchT39Bet~l{Z@0liTfkyJBCncq;0KU&EielQ!_^vql_}&$INMSAtF_Jlsvj2^U{J z8R%{X`LjRHpY0Wi{3L2%V=MdcQ^2Bg$bJV-ftfnkEfaZoQ(pc8iR(W%cfDn1$oy;_ zE+lD?Ep#psc_?5JGAnzo!XQ9?63da|hGd$JCcp-30rm!g+8t0sVS6`-h&o~yzt>ka z%`;oEK$b0J%1oKiy2m;m*^2(I%k0FvFtTmIk`jyWFaVe1BHv34uPr$>B^|_+#&j-{ z2yOzI!UOZ^tumQk#vD=ap>oU&g>hsJK6a||blXButKIbu_ZHozg>9~8mRMYEjd-!# zQ$~*@yC^P2;{WNCbOtaisO&SE2|-)%GD*{=`-PsXpx;6hhUYw)-lSc%S*L)`PN@d} zbekEF)ZEfiLE23j*bt6DI1_rYy668e!BhsW1kZco+tz7w4Tm zn0}axFy~;7!kmCP2vZ4~+9~{!I*0yJ>-ddh>(y8awV(ci-;-i(>?!>Ah+vYEezm{j z_k>vc!f*IJF4hKB;I~_>y-3{0;7`&W6>C4gfZrdH?mzhbL9EqY!|xHXR!)8oi?u_@ z@AqP@jQk!FYgOd;JF(VC$~!35x_-j%w_@!{{Kh}2wyw@&EOjuaW@)U; z3pCcFFxii4ti9mru-~J$Ud&Ki=iRNgR%St85&U{*Xsj0e=7Nu^24*?zWqBHF27W7H z=EE?#D1Vy9x*Fkaf6-X^SJhSp{4*X>TU$`pa_~C1U_OIi`~%Q^^t8s>g|O5qYU|?& z?|M~ZO?^;p-HP-s#4(h?4f{&y=qJBsk- zkEpE&<|EAvwbcOs2*mArM`I0}r?GO#_goJ2E+JnI{_V&!3~4@i3+=AdSofk|>)@_{ zc^v&@C_^8t&{!*A56jkA%h2B+pza6YUvr1rnhlfjuEuIEQCs^E=N!yVn5{5JVGh8o zhG~KEqAm)UooM6UhtXz~ZPTf(E8oyq<6)ogK>1dUwFT|zs?k^%{vGWcjs6c;TRE68 zUTs}oi@uIUS#Gp%B=kig-3JJp0DB8aIq{1%R`)+N);6R)@CxP#@|=KvaVN)4)R(LaDqFBk<9mv* z<~C%L8@Pl$qT?j=8tWj04`jt@C;+T+JRz=CAD#uc`IvYEX7*e?256~DfHwA|B6eM_ z$hmlf?oIYxHaw4UJS5gWBxLO$ckXR zYW)valUL|-t|m9>CyB|M$!E9TCMF-!E3PG<(cgbH`Kmrgglc8|l?y_7i2mNILV2iu zEWX3_cjFtT@56V5{(i&>*WZWlNcxS?{|f&|eJ{RI`f>1&*6+A1l*j1D!X8V%qxAQ{ zKAL`c{a+BO*L!~z$_2d%{&D&^*yHuqUZFfee;0fb$@3EBN%U*bkAXdzepB@66+-zP z`a9s0s!zdp4E^4zPlo+2`Zekeu&2?lNuLCJI{jwo6Jfube(%wXKMUn!_3`jA>(8S; zb zejObbVSkN&7kB&&`|I?(q~ikYcm)7oSI2pLt2!(fI?Jm&&cgl%{nm6$I@wwNX2)sx z{JrB8zW<-Z7h zdiq__@dNDtq~DbtM__-Ke*dTAFzo-LUvJ0vu>0uO-*E``RrI^M<2%?J==Z&jgRrlm z-^Pw_VP8wXO&$L_)mi?2M`>?o`3D`xf9))9?%0pe4?Fe^ILkLU4#57AvwV}|>w%6h z;j`JX4?Zo<@-2?N10B2Jv(@nxeAYS3w>iEX=-2_De>=L+>#ff6|2V!t(8td5cDyUn zTE5;{-r?9a;3;o&mUlWx%ufgL5B;43pu8P&tb3ZUYXs2s!SmBW>F!hC^z?)`K<5Iy z6NCbsMD-ibtYr_s40!kZE%5Gr0`EZLp!^99Wzo{v4RBQ7bcU8u_kSuQ2j{a&le};t zj2)%|<~0}>3|OVw!Dk6-8>mmvY4wt0?z4~1b}E!Wv8tvy6$&7LRhdqO95{Yenp1&? zD5~^Mh3q_A72;Gd*X30*Zh{eX+F1mp@*P8vN5R-#kXGY83O2V}#z5k{p64&LBNP6@ zMg0-c?vn9$;u@adr&TOxVuTPTJ%pff3aBhs*o6=~=!)sQ{(xwFF$7T&ZyYOlAq*SB zzo^rHB|@4P-i0rc9A0AeU1Fwol@!m+T3!#aq*IAUHf=jlbL0Ia{UmwXXCg>O#tYuG zPsw9glZHal62DZ6m0|9Z&ak)_&J|?jDo{%WKgNt^qq zcirDbn=;_cbcx1#?_|78!L(`Wf-CzW?+OV}qznjSYY_5>b5QKX*K*~Bs0Yr~TnfU8 zdL!C1cqWWljxe_r4wu<- zfjikaKWyvBXa7;&tirpRNwcWf>&hm)%Byuewgp($9tFr*y(G~gx#yjdil zZx34;L8WfER6763nY9O;sA}glT;8diN~L{^i~;}h7N=6-Q9=%cTIJ|y;uW;Qvo5=WCOf}bnd>6lRl^9!rEsn%bgjL3y1^BzIxvGAJLxs8zjr& zgK`7&wTeh#x!)okwkf!FspwlOO0KG4hlBD>W&03Tr{1G-2jry5lgSM&j(XBDV@C$z zE^;lQXOhW@@pn|<-8Q5EhrUzCh1!!BBVZWDfYA2Qbq4U#QR?W8zDxMiF-IAh9FV)I zQ#(q;y?)Hd76a=(Rl&PPI43gv-|=N*`rLb;ckg_jZ>xY%!bCZ`n0vH!qB>dDJyEfL zqOx~BB{O;{U2TPuF}>1CMHI)wU@-2iGDvbNSCQS zKr(F%F3IB4E{M>?0oPj6?RDz3vts2{W4%ry@ZefE?I%Pai*PVPH$WqC)1xjKGt?CY zcUkpY2%ld`s8+vObbN3cQJh6gEhAEjiy_$PGV+1dj1KI@IgP{l=rDds9?*wL6$4T_ zVNVOWO{Q7_@%uZOALKF}E2Vf_fRcDj`YnENw6u5opr!};;2ebN;5DtiE&`N5eHwLb zkk&+pjl}uwwm9h9JEgwOoiGR+eQTvL;RV?(z>$=s%a9jpHNi|6ei@mPQQ%h%gM7(C z-mJpVqsG2Ik^!g!0f-KGO57TZ2Y$)Jk$}Zo`~}i|tb%5++S5|t|N1?3GtuLyVR!jalDw8$Gf$sindbRNedJ3#`a@jr;N=b3*Im`?g!bRu8FXr>bHfetXJ=u4blv1ajsvVrR)am!#n8qOGbbj=s z$E$duis-*BVqtzG?*C{Q_3idv1U#d@ zj(s9G8n#9?FUmljg>@~>jte-rj@_hMjW^tI>>!ACk(_stb+l9M{eU=VS`u7% zTQInT1H{wXbt2X!V%}W;wI7SVk4YJ>;JtzMLL3?i?YU)#*K`0;g3R^S_AXib;Bc-1 zU6@86^ug1El~>`jQ%yv)yL3mTT9QWXRJ(6=Tnsom>S*a4#{8(1E!VO9Rxh82xT-i> z4(iwvDBSU;xb967^oNmfuONK^mLR%8kW|2o2nfL_y!J<2sdlc{SMI$n5?L(wbTJV( z)_1jVRKhukBv#EFE*Y~~h5PXDFuO^%Sf@PZoXW^D6YyIA6Axp7QG-VRIDQQ<*)XoZ zgWm$bsW35!Z^JK>#biH)Un9&^;?8EW%J7>36M{Io_?-t61G7?Ed$m0x2#x;tNVj6u zcZ=&zl8#ubc9ib$9i`o+@%>1C>ovX~$nPfdJK-8^ScHc~`xXMP03f~>ec#ih!I-wb zBQ$9+#Nk0-BoPRm5x;BIzHcSKiq!w5AsWnR-#0Wc7y#byC06Xi`^DO?X#{ocevu=3 zPtfgq#ax$+Ria2i6Xmy^YX%^tVB`ZwCpl)7Y!XAW|is8Ca{MmppRPg5R+F zDD08$+lpt2l2-M{q~C@o#9Ho;598uhQfN3Od14hv8wuU=(X?|C0D}!2rzwFb(%hkr zpt--3xu7ngupU=Q9{J!ANzMZ96U3yRcgp2B8DUCb&s`zxr8A|9b;=bcxze5<5oS+@>WuO*!~*7IpSbjUPd z=9cQo>JMz^h9hkU;fX8ca@-K4 zF{?cv{tT5@OISv7`35yasrQ;f4!@@VTy)G`oDzpX8*2~fHp(Hgt|eiS`hSZ|HWdS}VqIkWB<*pZrLj)aY5mj=Hfwn6aM;&_^}7_klNuFunEEQb0eIFi-iu;GT^ zQg=?7!!VVP7YS* zN2ITu(*4{@@jlk^*Z?=n?)1X9PJKi^2y3a+_Y-vB56-)0cAK0kI%qg2mRNU?n!iEl zr#+qIo8X`~eojhJEh69I7a4t~nD(vc$QnRy<47G0YHqmY#uIjGkfTf=kq*Xb_Dj*| z(s9F$f$pN7fKVPkgUJb36-n=6+n15aO5OUGOI>($g*-Gw#3(pQ#$7|J5ha@Ft&y3H zUT2Cc=f>mW(7)l^SKjW9KUIU)$B$bxtBN2&k$pQyI zIT$k>r^JrF|3SDx4K};>!0>b0Gztti4UvH1COL_ri3twmMeq%STH5IXxnoDBo`YGy zv58FQK+;rsU>QhB2kS)FuGdK02>O(7>+48z6W~w>ZqVt`1`g~^5Mp>4wwwDfJ>%B? zELKXW!;KxQeZaSt=WOFo=JUjk(&k&7UD+iT{?RW zFT|3bs~YNz)plxiDXxoMI9;v>_Ry*2xJ#W~7u(w_a2da&mEG@UV*2MFky+|PKn^iz z_hQmglE6VkIdks2xv{lWY>a2^ZM=I+NQ=GV zr)P$6VL01b>UChQb-%U@;?$(ITb#!}B=>=5&aU`pcNb)xv^0L3i8BuV{@UAnY7k?RH6 z9&DE|gfn$s1im#k;||^!Z$!=P6?Skje6%zMR3_e@E=$T#aL?ga{{{dbpR$G24ZO2- zEmlh7jP&h`L-0r|#xd=hsLei_4VTyAYDZ+7x-N9x7yUq2fh}i))OLgG0Q%X4P%=+# z>Xdq%Zd$T$1R;0Kqpx6o2g?OH3u6j6KU!bM+TU=dje(FvY~!Z!u8nuIOOhEY8^c|< zACcKJNIQ3`_onOib@y9l%Jz0aHxU760gZ1RIDDk*prep_Im-IYnihJfTzZ^rWM=iW z6JjoK!vib}@rkreqhToA{G z#YCqUG@x_xXAXdK4~Y!ww5|PvyJnyd8-Z`Nz@UwZp(7H7I8MFPldQi=0Hg~{{;RjE z7(G3364&!N00j{Kq+|x-kwG59U5)fa0rv-k`9vlP_vSkl$i>r03U&dfUJZDq(lYwr zL>veSFiWSvj0N^k`wInTQpd2UtNy%HJVqR;L57ybo2P4>8Nfb{!N9b(3r?Kys|Bg9 zjTLtH^LE4Y&MZdAlKZoW?oy~EW~xH!4V7XRk8LOX}WX{J`vb!K>Lk?XPx?%y@k3f&kSN6lJn5*&ds+wU%t(m zs<{n^nR>g^@3jvG(>=>^MD#JQ(Nb>9g^L=DM#H$2+ntkdcUIr#^ndi@ZF%`W`p+NS zuMWCt9mu(?C^0m0NfM@@qqo0uJuTBRNf-W{n#}cQP0i zzX8rEoD8T#qQC#+dgVAcU7B3F?ZPU(u>e?Din&_5^g6B@NQ4{5Z3(AYblvw0Aqm{- zAD@eJGK?t78EKxQ{fJYi{-B$Sl7xB&iCQEn$Pq_LsXCA`0Tu$QU!x*0C+(4EZY2@Y z{AORTEh7C2^zq5-F$9!TNK2|fnnWpLG1Gvf!jQUO7ipxM;ke!)<&yZmX(XpAvX!|s zTx8*`tw-Chk@U9z)pk(T=KZnNkljI>`@o;X)3%Fpr1f13(9tvieQ2R&a<5$?UTZRc z!x=7FF)l2#S1jB@_7InXnTdNQ9i{;;{#vkKO6^f(CSG^?$j};a5l5Pstf2MPH<0VI zTqzdm2N#Pn3MID$J4qc5-BJfW1(!p&AdsYe`=K<40_9-j7ATWY6%pnLzCJj1HGVu7 zOj-4o$QZ^~dFtjO#W<0;9il;PjsU-wq{nJ+ddNm6^lXF^a-{7S7s&7}F7?=PxI}V- z@whtOX!>S5*mu!!-q(O{8OLSXt9tDpJB{O<_G+iM-dUYKKRv|p3~ua5}$#CWk>k@Wxp~q26s{wVJz< zI&sZ7Sm$98*u2|2gBi|@YO-SE-v?w-us-3?z*>Ai$O*@k2_ z+4#uVT*Fb-rN+BFl~-%bTs0ae38r$0tCr&9r2{>l;E>~74cV}i@c)mu_W^6_$o_{X zH-AXNMFADjXiY8z1L`gWSJ2WmiByBytAGk1I1Jy2cD~p?XFl0SP+_I)$E9q}k zgI9}Q(BUmc9qRk3M_0GF35Q80)T&nek0OevVTcLJva}?wP2e(=h(mWE8zq^ON(&6P08n1A zRpu|R`)Pdn-_q#1OJmL)AB`^xo)I+&u7&PDHV&O+bd z?(Z8%hWp0j{y2ujIQVgQk?7?v^d0*T{X>52zwL+qsvrA1f9NmyvHxkGKja25ZhKXP z@={{KZ?JFGI#O`jD1-mU0JlFLUtaldvxYl5r4tK(l)zQ~YL;koXIMN2otTZLs*2e_ zaD-`b-xe7{Q`^A52FL))qtJJSrxp+iFaQQ{Pcs}(lUs_j9mp;K$^lkD17H_mFJKWM z6R;4FP#%~Rj=0Euvy)dOn9hRsYimx-J_EmtfaFUFrc!I7>9m}f z7B;d;o8N(aBc|rBa~A(%ULV3FB2Fvf%mCgcKx`;vY#Wr8@ORmym3SuNxeLz>JSRWECY9nj z{cXt7;yDxFPXdAwMeE{#5@xB!}ERHCz{@Ef-G^f(sBo| zt3_$)fuG^e$UoxjMBcRsHvzC2xJ}Lzvq$p@rh|ZPl-W6Zf=NP5+GB9P{4iKKN@6UR9%L+~fD*hw2wsqA`UjpfU;Hu=Doc|Val}M`LrA14JQ6ay$JnHNfD+IScbzhnT*Gr= z3!AhY(2cMXE!q;^``$+#0h09y_W|N)MLhvF0YZTj1{e=`06@dgu%+-j)!yFDwy{YA z$JwOa%L9>i=WIdv_W2UT(-494YCs9-`WP@B?*pJ~4Zw}}2?&=2*b8_E@PA9E9e&o6 zY*HeCrjdc?Ccs%hHwAw_lpnH5+uD(509a*MMzTpUfGGed!w|#Lvim9C%SBqdXVdaK z+eFI@moLc#lWop#viBEu%znEqbGxCqb@rWSUt2UP`J>s(LGLylF_{RPsYE|tcJ{>V zhW|m>X9JU_qaNbWm(Sr{@myfi48StL7QjA02cUbu+A@UaWhwf_1=Q=61e1GAqG|GP z1C##eS8UQFcrF3h0Qo}+rgyp$OwO+pOq*4SCK>v^6*w7ziKa<-ZUoK?fKz~rprQR@ zf~g13+xV7m4ouqoRe~vL(}~&VdjgYsOW34|y|iA~%%tlHrfuIOnBLf&U}^wF!hPfp z^o{4x*Cr&I%y>TyJ&F(fE-*=^BxDx+R|EC{jswu`-X3u)FewqR9PoR90bzRnptfuV zY}=4vx`gyj0vZ7a;eQFB{XQ_M2H&}Omg8B8=Y+)xrf2b<0~mn|$;o&Ua^gKfP>?U+ zuj&KM_Jk(k`DX~xcE6T@@6#A#ryfl(#Q=9LzAFHK1#EssVp;jy1d|Hx1Ne@<9hhW4 zl3@CFJH|4=6Zj5C-ap6tUjXj|$U2f)YA4vg4GO!&r||7`JSPCA0wSrKL=a-}Gyt^p z`*|YttUWRN8lW3+88Fq%W>#e~nMxCzxe{vrm0~P%hT{P0N z2~2uwBAfJU0O^)ycAr#RGCx#X+<;C%Vmsh3YKtAuc0lqexC1iLJ{thX@V*UCf^Z*W zEXRnKDNGs=4lWE#I-HEYfM@r-L{r8YwM982(bV%G=7YpUlNRq^!SB_oz@&fT*$02g zBY{aDKFlT=@k{}10UQC;!cG2*z@!HNb&m!neF<>D?IgYm<*Uxu}3KD9@>1BZes7U7|O2EqcNu{Y?Hu$ zJ?Lka`S=c9Ln3eN-k#pK&%&aIKQ~yC$Bsil%1y%~S~+)+@^bmV;U>A?S=_IF>|-Jz zq;LOvW-xPcHe^ekOmVzvXOrm#ci}Dtt6cH3po!fLviS!Ns~02}^a~{RC3lfZvshGr zPUC%l&>skqr(lCb-QN3A+(!nT^!!o0R0@Ae=b`@#oz$=H|Ki6V{9(Lh-ofEx_m31Q zVk3dcY`AILRvi4s?o5W-qdm^D7YA+sOCv5)9bL~VV~u9KJLVB>y!%75Hn%CkMLAVm z0L>!eYh>7kP@Pgsui}==VhZ3V@}C?5WC6E;Dj)R)pL@e+$MM;_`PnGuR8_XYmzb}9 z%kl?@Mc;mXzdQdM5nGvSRPk%?wG=TA>sJr11WnJW$~8cW0OiE0^_y#O z88U6KIc?DM*kHYTG5^>g!a?xb;T8%j5*+S$hgt0~2RdY|!^}ElK@M|}LpH);9^sIU zbeKoBc$^m=9_(mVH?`QQgj!d?LAp6vU^}m%R+4KN{VQ*#9~o*0343aYUGt`g|0SiX zfR>w1vVP{EwuN$raQ!s)BkgtBpnTySZ}SA|V$)QUY|N6lRpw_?af?^NJ~Ga8xU8bC>$YgQgQhd-Kc%A=`Jw-k&Zz&V z>HIf(a=!P`^WX5GH;@(~>I*Xb{4Hu}gugL>J4<``?g6KAuYtcw-M}BXKH#}N07tPv zvLPeNJ%IHdl@%5HY8;!Lo~JPH`&H6Os;na&Xa}U{o}=38rq+9iPha_)f2TT1C@&eU zzQvuyKUE7nU4M?jbBglvz*d{R^8>Ca+mIfU+q;@T`oCE?di@(^iJ5EskjW=H~xF=2q^`Vg`?K! z5tjcJn_#E~n_My0<%U3o%PMZ&Z=zg$TKf92gQI!$YUzHiLzG9o_kHF^-$UYl5B^@a zc*XrZ;olf&eq*5Sje#QZLMRp_V*j9#jU-!`=GO)q0?xcPP(JqLQOEJq2KDC^$B#0u z6O#6Ef11LIHg|JYfYq!BAPpIsS3X@p0$(XD&8)~MA0JpzF}^TV8f`678>)Q$(Gv2D;Y zT|1zAY5*3YOWOFS2D}D>fB5gMv}4x1Grl2RJuhu8c|R8zHxIIDpW&1a;nBTp?|1ALmcQ8U9>XH?24ek_fG zRVq+*tN}~`!dk&gZZkG;=Os`-K{NIZG-Zfo$dgUR&3*R8v1^$W^U{sTIKAPV^YXX` z$rknAH!%^=8t4;Rnu8lf@svH|pxt8i;9SJ&|5%JX`DG-$b=FIlp4dO402HMpA@u-oB zeO)Brw$P|PFFysn6qLG)zVV9ACt;4^PWhnZfUF)>;}$gU7ivZ>_}C{G1&*$0%jv*q z7fb5hf-8Bn<{+y)G%m1FG$3AxDlRTAn(l!7kHnEYMpI2wC=e?i?#U?g3uNXQ@V4q^ zdcgH;eS?&}bBBNX&YqVlE)?%_*@6oXE1VZJPKRxT^8!1r-RjT<|H%L~n$9y&)8uID zx5Iky3uYWAIBX#$mmRIAGs@F@9}q8|>;LGpBCP+ZAQMF#uXwMX{tS!U0`Yk*wNf3q5>;7V|Zs z;l?R-N3&LxQ_}y_|2e_;c~z)y4hLH0Mr8F#byb;57h3HsoNlP%@pva9V||Gp_aGKK zZ6(e?XK~6keL*52VO>Ei!^&8LB7r=qf)atBy?}2yT*3>Cs8g4MI9jy%&|@4qa%Bb?#>Z>wu{T z42)+2R&h~=Zs&z_8K}@(hG1uNR_p1a0%u)xZpwzDeM??uv*u0#Ucwbs`Gxir7iv8^ zI5%7dwm2c5X3_DZYXpG>=7MchNZ+v1=`3pg^deKQV^Pg_>$vhG!9wd;9B}j>3XK82 zf`R%Ml;MGL+;ExVt8kU%6vI9!zE>(G&;w6kH4>BXS)f_m?>L z6zmY`LRdrkiJJv&;XeT?0MD73Vx8QaL0>I3TV) z5f)B4L*iaxh#v3#JKRAL*YSSSpSXakO*9TsJilBSc%V^rN?pp1YPn2aWe@HB=3I<3 z!}z|oB(X*c62yjHxt5OLpp$EHQ%37dOG+BeYoBl+i$>Ke@E7|J+e7kH}9mWNbM`lJHDP*SxB>1iRB?p53e~d3+3aT zYA9zM|86-Z&xU?Xo(=sckDjiu-+l<>9v&9(h?wf*860cKkul8C!Hzo$04Fy>yRijNVW ze+_bHc+KYj9>RAkjFO92Hl7pf>v^3VKQ5xX}>$Ev#Vb-Z8K#FS4&LSJXG4l4&S(3~*A;a5Z=_AcKR0Y)=Di$Lg~VCfCZKWWTujyE>k234 z38xcVw)=5w7++{0R{jL*t!e7^{;J79RG*SLSKh@G+kkJ))`db_*x$Ucziy$wK0Zei zq2=%QpH29m_Ba2uzwW1g|Fd`f#j)ujg3-(UoH)j1ljfiI*ZsWT*JD2sd@XRcud?)4 zhvTK;a%SBNSD8hcjs0~-U$~qOU%0LZec>vkYr9d2)EC%*ZV8*V?oa>)59;Yaj0kLa%(;VXx( z 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 z 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 -#include -#include -#include -#include -#include -#ifdef DISPLAY_ENABLED -#define SCREEN_WIDTH 128 -#define SCREEN_HEIGHT 64 -#define OLED_RESET -1 -#include -#include -Adafruit_SSD1306 display(SCREEN_WIDTH, SCREEN_HEIGHT, &Wire, OLED_RESET); -#include -Adafruit_NeoPixel rgb_led_1 = Adafruit_NeoPixel(1, 1, NEO_GRB + NEO_KHZ800); - -#endif -#include "esp_partition.h" -#include "esp_ota_ops.h" -#include "esp_system.h" - -String ssid; -uint8_t mac[6]; - -// Create an instance of the server -WebServer server(80); -bool displayEnabled; - -const int BUTTON_PIN = 0; // GPIO for the button -volatile unsigned long lastPressTime = 0; // Time of last button press -volatile bool doublePressDetected = false; // Flag for double press -const unsigned long doublePressInterval = 500; // Max. time (in ms) between two presses for double press -volatile int pressCount = 0; // Counts the button presses - -const unsigned char epd_bitmap_wifi[] PROGMEM = { - 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, - 0x00, 0x00, 0x1f, 0x80, 0x00, 0x00, 0x00, 0x01, 0xff, 0xf0, 0x00, 0x00, 0x00, 0x07, 0xff, 0xfc, 0x00, 0x00, 0x00, 0x1f, 0xe0, 0xff, 0x00, 0x00, - 0x00, 0x3f, 0x00, 0x0f, 0x80, 0x00, 0x00, 0x7c, 0x00, 0x03, 0xe0, 0x00, 0x00, 0xf0, 0x00, 0x01, 0xf0, 0x00, 0x01, 0xe0, 0x00, 0x00, 0x78, 0x00, - 0x03, 0xc0, 0x00, 0x00, 0x38, 0x00, 0x07, 0x80, 0x00, 0x00, 0x1c, 0x00, 0x0f, 0x00, 0x06, 0x00, 0x0e, 0x00, 0x0e, 0x00, 0x7f, 0xe0, 0x0e, 0x00, - 0x0c, 0x01, 0xff, 0xf0, 0x06, 0x00, 0x00, 0x07, 0xff, 0xfc, 0x02, 0x00, 0x00, 0x0f, 0x80, 0x3e, 0x00, 0x00, 0x00, 0x1f, 0x00, 0x0f, 0x00, 0x00, - 0x00, 0x1c, 0x00, 0x07, 0x80, 0x00, 0x00, 0x38, 0x00, 0x03, 0xc0, 0x00, 0x00, 0x70, 0x00, 0x01, 0xc0, 0x00, 0x00, 0x70, 0x00, 0x00, 0xc0, 0x00, - 0x00, 0x20, 0x00, 0x00, 0x40, 0x00, 0x00, 0x00, 0x1f, 0x80, 0x00, 0x00, 0x00, 0x00, 0x7f, 0xc0, 0x00, 0x00, 0x00, 0x00, 0xff, 0xe0, 0x00, 0x00, - 0x00, 0x01, 0xe0, 0xf0, 0x00, 0x00, 0x00, 0x01, 0xc0, 0x78, 0x00, 0x00, 0x00, 0x03, 0x80, 0x38, 0x00, 0x00, 0x00, 0x01, 0x00, 0x10, 0x00, 0x00, - 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, - 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x06, 0x00, 0x00, 0x00, 0x00, 0x00, 0x06, 0x00, 0x00, 0x00, 0x00, 0x00, 0x04, 0x00, 0x00, 0x00, - 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 -}; - -// 'checkmark', 44x44px -const unsigned char epd_bitmap_checkmark[] PROGMEM = { - 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, - 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, - 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, - 0x00, 0x00, 0x00, 0x01, 0x80, 0x00, 0x00, 0x00, 0x00, 0x03, 0x80, 0x00, 0x00, 0x00, 0x00, 0x07, 0x80, 0x00, 0x00, 0x00, 0x00, 0x0f, 0x00, 0x00, - 0x00, 0x00, 0x00, 0x1e, 0x00, 0x00, 0x00, 0x00, 0x00, 0x3c, 0x00, 0x00, 0x00, 0x00, 0x00, 0x78, 0x00, 0x00, 0x00, 0x0e, 0x00, 0xf0, 0x00, 0x00, - 0x00, 0x0f, 0x01, 0xe0, 0x00, 0x00, 0x00, 0x0f, 0x83, 0xc0, 0x00, 0x00, 0x00, 0x07, 0xc7, 0x80, 0x00, 0x00, 0x00, 0x03, 0xef, 0x00, 0x00, 0x00, - 0x00, 0x01, 0xfe, 0x00, 0x00, 0x00, 0x00, 0x00, 0xfc, 0x00, 0x00, 0x00, 0x00, 0x00, 0x78, 0x00, 0x00, 0x00, 0x00, 0x00, 0x10, 0x00, 0x00, 0x00, - 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, - 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, - 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, - 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 -}; - -void IRAM_ATTR handleButtonPress() { - unsigned long currentTime = millis(); // Get current time - - // Debounce: If the current press is too close to the last one, ignore it - if (currentTime - lastPressTime > 50) { - pressCount++; // Count the button press - - // Check if this is the second press within the double-press interval - if (pressCount == 2 && (currentTime - lastPressTime <= doublePressInterval)) { - doublePressDetected = true; // Double press detected - pressCount = 0; // Reset counter - } - - lastPressTime = currentTime; // Update the time of the last press - } -} - -// Function to switch the boot partition to OTA1 -void setBootPartitionToOTA0() { - const esp_partition_t *ota0_partition = esp_partition_find_first(ESP_PARTITION_TYPE_APP, ESP_PARTITION_SUBTYPE_APP_OTA_0, NULL); - - if (ota0_partition) { - // Set OTA1 as new boot partition - esp_ota_set_boot_partition(ota0_partition); - Serial.println("Boot partition changed to OTA0. Restarting..."); - - // Restart to boot from the new partition - esp_restart(); - } else { - Serial.println("OTA1 partition not found!"); - } -} - -void setupDisplay() { - displayEnabled = display.begin(SSD1306_SWITCHCAPVCC, 0x3D); - if (displayEnabled) { - display.display(); - delay(100); - display.clearDisplay(); - } -} - -void displayStatusBar(int progress) { - display.clearDisplay(); - display.setCursor(24, 8); - display.println("Sketch wird"); - display.setCursor(22, 22); - display.println("hochgeladen!"); - - display.fillRect(0, SCREEN_HEIGHT - 24, SCREEN_WIDTH - 4, 8, BLACK); // Clear status bar area - display.drawRect(0, SCREEN_HEIGHT - 24, SCREEN_WIDTH - 4, 8, WHITE); // Draw border - int filledWidth = (progress * SCREEN_WIDTH - 4) / 100; // Calculate progress width - display.fillRect(1, SCREEN_HEIGHT - 23, filledWidth - 4, 6, WHITE); // Fill progress bar - - display.setCursor((SCREEN_WIDTH / 2) - 12, SCREEN_HEIGHT - 10); - display.setTextSize(1); - display.setTextColor(WHITE, BLACK); - display.print(progress); - display.println(" %"); - display.display(); -} - -void displayWelcomeScreen() { - display.clearDisplay(); - - // Draw WiFi symbol - display.drawBitmap(0, 12, epd_bitmap_wifi, 44, 44, WHITE); - - // Display SSID text - display.setCursor(40, 13); - display.setTextSize(1); - display.setTextColor(WHITE, BLACK); - display.println("Verbinde dich"); // "Connect" - display.setCursor(60, 27); - display.println("mit:"); // "with" - - // Display SSID - display.setCursor(40, 43); - display.setTextSize(1); // Larger text for SSID - display.print(ssid); - - display.display(); -} - -void displaySuccessScreen() { - display.clearDisplay(); - - // Draw WiFi symbol - display.drawBitmap(0, 12, epd_bitmap_checkmark, 44, 44, WHITE); - - // Display SSID text - display.setCursor(48, 22); - display.setTextSize(1); - display.setTextColor(WHITE, BLACK); - display.println("Erfolgreich"); // "Successfully" - display.setCursor(48, 36); - display.println("hochgeladen!"); // "uploaded!" - - display.display(); -} - -void wipeDisplay() { - display.clearDisplay(); - display.println(""); - display.display(); -} - -void setupWiFi() { - WiFi.macAddress(mac); - char macLastFour[5]; - snprintf(macLastFour, sizeof(macLastFour), "%02X%02X", mac[4], mac[5]); - ssid = "senseBox:" + String(macLastFour); - - // Define the IP address, gateway, and subnet mask - IPAddress local_IP(192, 168, 1, 1); // The new IP address - IPAddress gateway(192, 168, 1, 1); // Gateway address (can be the same as the AP's IP) - IPAddress subnet(255, 255, 255, 0); // Subnet mask - - // Set the IP address, gateway, and subnet mask of the access point - WiFi.softAPConfig(local_IP, gateway, subnet); - - // Start the access point - WiFi.softAP(ssid.c_str()); -} - -void setupOTA() { - // Handle updating process - server.on( - "/sketch", HTTP_POST, - []() { - server.sendHeader("Connection", "close"); - server.send(200, "text/plain", (Update.hasError()) ? "FAIL" : "OK"); - ESP.restart(); - }, - []() { - HTTPUpload &upload = server.upload(); - - if (upload.status == UPLOAD_FILE_START) { - Serial.setDebugOutput(true); - size_t fsize = UPDATE_SIZE_UNKNOWN; - if (server.clientContentLength() > 0) { - fsize = server.clientContentLength(); - } - Serial.printf("Receiving Update: %s, Size: %d\n", upload.filename.c_str(), fsize); - - Serial.printf("Update: %s\n", upload.filename.c_str()); - if (!Update.begin(fsize)) { //start with max available size - Update.printError(Serial); - } - } else if (upload.status == UPLOAD_FILE_WRITE) { - /* flashing firmware to ESP*/ - if (Update.write(upload.buf, upload.currentSize) != upload.currentSize) { - Update.printError(Serial); - } else { - int progress = (Update.progress() * 100) / Update.size(); - displayStatusBar(progress); // Update progress on status bar - } - } else if (upload.status == UPLOAD_FILE_END) { - if (Update.end(true)) { //true to set the size to the current progress - displaySuccessScreen(); - delay(3000); - wipeDisplay(); - } else { - Update.printError(Serial); - } - Serial.setDebugOutput(false); - } - yield(); - } - ); -} - -void setup() { - // Start Serial communication - Serial.begin(115200); - rgb_led_1.begin(); - rgb_led_1.setBrightness(30); - rgb_led_1.setPixelColor(0, rgb_led_1.Color(51, 51, 255)); - rgb_led_1.show(); - - // Configure button pin as input - pinMode(BUTTON_PIN, INPUT_PULLUP); - - // Interrupt for the button - attachInterrupt(digitalPinToInterrupt(BUTTON_PIN), handleButtonPress, FALLING); - -#ifdef DISPLAY_ENABLED - setupDisplay(); -#endif - setupWiFi(); - // Set the ESP32 as an access point - setupOTA(); - server.begin(); -} - -void loop() { - // Handle client requests - server.handleClient(); - -#ifdef DISPLAY_ENABLED - displayWelcomeScreen(); -#endif - - if (doublePressDetected) { - Serial.println("Doppeldruck erkannt!"); // "Double press detected!" - setBootPartitionToOTA0(); -#ifdef DISPLAY_ENABLED - display.setCursor(0, 0); - display.setTextSize(1); - display.setTextColor(WHITE, BLACK); - display.println(""); - display.display(); - delay(50); -#endif - // Restart to boot from the new partition - esp_restart(); - } -} diff --git a/variants/sensebox_eye/variant.cpp b/variants/sensebox_eye/variant.cpp index 1eef839494e..d8a37f58758 100644 --- a/variants/sensebox_eye/variant.cpp +++ b/variants/sensebox_eye/variant.cpp @@ -1,34 +1,6 @@ -/* - * The MIT License (MIT) - * - * Copyright (c) 2021 Ha Thach (tinyusb.org) for Adafruit Industries - * - * Permission is hereby granted, free of charge, to any person obtaining a copy - * of this software and associated documentation files (the "Software"), to deal - * in the Software without restriction, including without limitation the rights - * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell - * copies of the Software, and to permit persons to whom the Software is - * furnished to do so, subject to the following conditions: - * - * The above copyright notice and this permission notice shall be included in - * all copies or substantial portions of the Software. - * - * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR - * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, - * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE - * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER - * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, - * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN - * THE SOFTWARE. - */ - #include "esp32-hal-gpio.h" #include "pins_arduino.h" #include "driver/rmt_tx.h" -#include "esp_log.h" -#include "esp_partition.h" -#include "esp_system.h" -#include "esp_ota_ops.h" extern "C" { @@ -72,29 +44,7 @@ void initVariant(void) { .loop_count = 0 }; - // define button pin - pinMode(47, INPUT_PULLUP); - - // Check if button is pressed - if (digitalRead(47) == LOW) { - // When the button is pressed and then released, boot into the OTA1 partition - const esp_partition_t *ota1_partition = esp_partition_find_first(ESP_PARTITION_TYPE_APP, ESP_PARTITION_SUBTYPE_APP_OTA_1, NULL); - - if (ota1_partition) { - esp_err_t err = esp_ota_set_boot_partition(ota1_partition); - if (err == ESP_OK) { - uint8_t pixel[3] = { 0x00, 0x00, 0x10 }; // blue - blinkLED(pixel, led_chan, ws2812_encoder, tx_config); - esp_restart(); // restart, to boot OTA1 partition - } else { - uint8_t pixel[3] = { 0x00, 0x10, 0x00 }; // red - blinkLED(pixel, led_chan, ws2812_encoder, tx_config); - ESP_LOGE("OTA", "Error setting OTA1 partition: %s", esp_err_to_name(err)); - } - } - } else { - uint8_t pixel[3] = { 0x10, 0x00, 0x00 }; // green - blinkLED(pixel, led_chan, ws2812_encoder, tx_config); - } + uint8_t pixel[3] = { 0x10, 0x00, 0x00 }; // green + blinkLED(pixel, led_chan, ws2812_encoder, tx_config); } } From 28e90766018ffe46fef16526487ba2a905052e53 Mon Sep 17 00:00:00 2001 From: Paula Scharf Date: Sun, 20 Jul 2025 22:53:35 +0200 Subject: [PATCH 129/173] feat(board): build name uppercase --- boards.txt | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/boards.txt b/boards.txt index 2363fe2a63b..07c34649ea0 100644 --- a/boards.txt +++ b/boards.txt @@ -41343,7 +41343,7 @@ sensebox_eye.build.target=esp32s3 sensebox_eye.build.mcu=esp32s3 sensebox_eye.build.core=esp32 sensebox_eye.build.variant=sensebox_eye -sensebox_eye.build.board=sensebox_eye +sensebox_eye.build.board=SENSEBOX_EYE sensebox_eye.build.usb_mode=0 sensebox_eye.build.cdc_on_boot=1 From 349b11d2d86b37f3e6613cf25ea490b02c9fdecf Mon Sep 17 00:00:00 2001 From: Paula Scharf Date: Sun, 20 Jul 2025 23:00:05 +0200 Subject: [PATCH 130/173] feat(board): name tag lowercase --- boards.txt | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/boards.txt b/boards.txt index 07c34649ea0..1043ece4445 100644 --- a/boards.txt +++ b/boards.txt @@ -41313,12 +41313,12 @@ sensebox_mcu_esp32s2.menu.EraseFlash.all.upload.erase_cmd=-e # senseBox Eye sensebox_eye.name=senseBox Eye -senseBox_eye.vid.0=0x303A -senseBox_eye.pid.0=0x82D1 -senseBox_eye.vid.1=0x303A -senseBox_eye.pid.1=0x82D2 -senseBox_eye.vid.2=0x303A -senseBox_eye.pid.2=0x82D3 +sensebox_eye.vid.0=0x303A +sensebox_eye.pid.0=0x82D1 +sensebox_eye.vid.1=0x303A +sensebox_eye.pid.1=0x82D2 +sensebox_eye.vid.2=0x303A +sensebox_eye.pid.2=0x82D3 sensebox_eye.bootloader.tool=esptool_py sensebox_eye.bootloader.tool.default=esptool_py From cb3329be60f7cb446faa4aaa4572975ffdabab42 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jan=20Proch=C3=A1zka?= <90197375+P-R-O-C-H-Y@users.noreply.github.com> Date: Mon, 21 Jul 2025 14:47:21 +0200 Subject: [PATCH 131/173] feat(zigbee): Add callback option for default response message (#11613) * feat(zigbee): Add cb option for default response message * fix(example): Add timeout and fix spelling * feat(zigbee): Add global default response cb option * fix(example): Use task for measure and sleep * fix(zigbee): Remove debug logs * ci(pre-commit): Apply automatic fixes * fix(example): Add retry and fix typo * ci(pre-commit): Apply automatic fixes --------- Co-authored-by: pre-commit-ci-lite[bot] <117423508+pre-commit-ci-lite[bot]@users.noreply.github.com> --- .../Zigbee_Temp_Hum_Sensor_Sleepy.ino | 81 ++++++++++++++++--- libraries/Zigbee/keywords.txt | 2 + libraries/Zigbee/src/Zigbee.h | 3 + libraries/Zigbee/src/ZigbeeCore.cpp | 7 ++ libraries/Zigbee/src/ZigbeeCore.h | 12 +++ libraries/Zigbee/src/ZigbeeEP.cpp | 12 ++- libraries/Zigbee/src/ZigbeeEP.h | 15 +++- libraries/Zigbee/src/ZigbeeHandlers.cpp | 10 +++ libraries/Zigbee/src/ZigbeeTypes.h | 30 +++++++ 9 files changed, 158 insertions(+), 14 deletions(-) create mode 100644 libraries/Zigbee/src/ZigbeeTypes.h diff --git a/libraries/Zigbee/examples/Zigbee_Temp_Hum_Sensor_Sleepy/Zigbee_Temp_Hum_Sensor_Sleepy.ino b/libraries/Zigbee/examples/Zigbee_Temp_Hum_Sensor_Sleepy/Zigbee_Temp_Hum_Sensor_Sleepy.ino index e9d08d32175..54c085fbfea 100644 --- a/libraries/Zigbee/examples/Zigbee_Temp_Hum_Sensor_Sleepy/Zigbee_Temp_Hum_Sensor_Sleepy.ino +++ b/libraries/Zigbee/examples/Zigbee_Temp_Hum_Sensor_Sleepy/Zigbee_Temp_Hum_Sensor_Sleepy.ino @@ -32,18 +32,49 @@ #include "Zigbee.h" +#define USE_GLOBAL_ON_RESPONSE_CALLBACK 1 // Set to 0 to use local callback specified directly for the endpoint. + /* Zigbee temperature + humidity sensor configuration */ #define TEMP_SENSOR_ENDPOINT_NUMBER 10 #define uS_TO_S_FACTOR 1000000ULL /* Conversion factor for micro seconds to seconds */ #define TIME_TO_SLEEP 55 /* Sleep for 55s will + 5s delay for establishing connection => data reported every 1 minute */ +#define REPORT_TIMEOUT 1000 /* Timeout for response from coordinator in ms */ uint8_t button = BOOT_PIN; ZigbeeTempSensor zbTempSensor = ZigbeeTempSensor(TEMP_SENSOR_ENDPOINT_NUMBER); +uint8_t dataToSend = 2; // Temperature and humidity values are reported in same endpoint, so 2 values are reported +bool resend = false; + +/************************ Callbacks *****************************/ +#if USE_GLOBAL_ON_RESPONSE_CALLBACK +void onGlobalResponse(zb_cmd_type_t command, esp_zb_zcl_status_t status, uint8_t endpoint, uint16_t cluster) { + Serial.printf("Global response command: %d, status: %s, endpoint: %d, cluster: 0x%04x\r\n", command, esp_zb_zcl_status_to_name(status), endpoint, cluster); + if ((command == ZB_CMD_REPORT_ATTRIBUTE) && (endpoint == TEMP_SENSOR_ENDPOINT_NUMBER)) { + switch (status) { + case ESP_ZB_ZCL_STATUS_SUCCESS: dataToSend--; break; + case ESP_ZB_ZCL_STATUS_FAIL: resend = true; break; + default: break; // add more statuses like ESP_ZB_ZCL_STATUS_INVALID_VALUE, ESP_ZB_ZCL_STATUS_TIMEOUT etc. + } + } +} +#else +void onResponse(zb_cmd_type_t command, esp_zb_zcl_status_t status) { + Serial.printf("Response command: %d, status: %s\r\n", command, esp_zb_zcl_status_to_name(status)); + if (command == ZB_CMD_REPORT_ATTRIBUTE) { + switch (status) { + case ESP_ZB_ZCL_STATUS_SUCCESS: dataToSend--; break; + case ESP_ZB_ZCL_STATUS_FAIL: resend = true; break; + default: break; // add more statuses like ESP_ZB_ZCL_STATUS_INVALID_VALUE, ESP_ZB_ZCL_STATUS_TIMEOUT etc. + } + } +} +#endif + /************************ Temp sensor *****************************/ -void meausureAndSleep() { +static void meausureAndSleep(void *arg) { // Measure temperature sensor value float temperature = temperatureRead(); @@ -55,13 +86,35 @@ void meausureAndSleep() { zbTempSensor.setHumidity(humidity); // Report temperature and humidity values - zbTempSensor.report(); + zbTempSensor.report(); // reports temperature and humidity values (if humidity sensor is not added, only temperature is reported) Serial.printf("Reported temperature: %.2f°C, Humidity: %.2f%%\r\n", temperature, humidity); - // Add small delay to allow the data to be sent before going to sleep - delay(100); + unsigned long startTime = millis(); + const unsigned long timeout = REPORT_TIMEOUT; + + Serial.printf("Waiting for data report to be confirmed \r\n"); + // Wait until data was successfully sent + int tries = 0; + const int maxTries = 3; + while (dataToSend != 0 && tries < maxTries) { + if (resend) { + Serial.println("Resending data on failure!"); + resend = false; + dataToSend = 2; + zbTempSensor.report(); // report again + } + if (millis() - startTime >= timeout) { + Serial.println("\nReport timeout! Report Again"); + dataToSend = 2; + zbTempSensor.report(); // report again + startTime = millis(); + tries++; + } + Serial.printf("."); + delay(50); // 50ms delay to avoid busy-waiting + } - // Put device to deep sleep + // Put device to deep sleep after data was sent successfully or timeout Serial.println("Going to sleep now"); esp_deep_sleep_start(); } @@ -92,6 +145,16 @@ void setup() { // Add humidity cluster to the temperature sensor device with min, max and tolerance values zbTempSensor.addHumiditySensor(0, 100, 1); + // Set callback for default response to handle status of reported data, there are 2 options. + +#if USE_GLOBAL_ON_RESPONSE_CALLBACK + // Global callback for all endpoints with more params to determine the endpoint and cluster in the callback function. + Zigbee.onGlobalDefaultResponse(onGlobalResponse); +#else + // Callback specified for endpoint + zbTempSensor.onDefaultResponse(onResponse); +#endif + // Add endpoint to Zigbee Core Zigbee.addEndpoint(&zbTempSensor); @@ -117,8 +180,8 @@ void setup() { Serial.println(); Serial.println("Successfully connected to Zigbee network"); - // Delay approx 1s (may be adjusted) to allow establishing proper connection with coordinator, needed for sleepy devices - delay(1000); + // Start Temperature sensor reading task + xTaskCreate(meausureAndSleep, "temp_sensor_update", 2048, NULL, 10, NULL); } void loop() { @@ -141,7 +204,5 @@ void loop() { } } } - - // Call the function to measure temperature and put the device to sleep - meausureAndSleep(); + delay(100); } diff --git a/libraries/Zigbee/keywords.txt b/libraries/Zigbee/keywords.txt index 23f3af3bf02..68721c1a66f 100644 --- a/libraries/Zigbee/keywords.txt +++ b/libraries/Zigbee/keywords.txt @@ -47,6 +47,7 @@ zb_power_source_t KEYWORD1 ZigbeeWindowCoveringType KEYWORD1 ZigbeeFanMode KEYWORD1 ZigbeeFanModeSequence KEYWORD1 +zb_cmd_type_t KEYWORD1 ####################################### # Methods and Functions (KEYWORD2) @@ -96,6 +97,7 @@ getTime KEYWORD2 getTimezone KEYWORD2 addOTAClient KEYWORD2 clearBoundDevices KEYWORD2 +onDefaultResponse KEYWORD2 # ZigbeeLight + ZigbeeColorDimmableLight onLightChange KEYWORD2 diff --git a/libraries/Zigbee/src/Zigbee.h b/libraries/Zigbee/src/Zigbee.h index 65c9e7f0daa..ab94a163f3e 100644 --- a/libraries/Zigbee/src/Zigbee.h +++ b/libraries/Zigbee/src/Zigbee.h @@ -2,6 +2,9 @@ #pragma once +// Common types and functions +#include "ZigbeeTypes.h" + // Core #include "ZigbeeCore.h" #include "ZigbeeEP.h" diff --git a/libraries/Zigbee/src/ZigbeeCore.cpp b/libraries/Zigbee/src/ZigbeeCore.cpp index c49dedb221f..bf652642ac1 100644 --- a/libraries/Zigbee/src/ZigbeeCore.cpp +++ b/libraries/Zigbee/src/ZigbeeCore.cpp @@ -32,6 +32,7 @@ ZigbeeCore::ZigbeeCore() { _scan_duration = 3; // default scan duration _rx_on_when_idle = true; _debug = false; + _global_default_response_cb = nullptr; // Initialize global callback to nullptr if (!lock) { lock = xSemaphoreCreateBinary(); if (lock == NULL) { @@ -792,6 +793,12 @@ const char *ZigbeeCore::getDeviceTypeString(esp_zb_ha_standard_devices_t deviceI } } +void ZigbeeCore::callDefaultResponseCallback(zb_cmd_type_t resp_to_cmd, esp_zb_zcl_status_t status, uint8_t endpoint, uint16_t cluster) { + if (_global_default_response_cb) { + _global_default_response_cb(resp_to_cmd, status, endpoint, cluster); + } +} + ZigbeeCore Zigbee = ZigbeeCore(); #endif // CONFIG_ZB_ENABLED diff --git a/libraries/Zigbee/src/ZigbeeCore.h b/libraries/Zigbee/src/ZigbeeCore.h index 69c91c63ac4..df334e1620d 100644 --- a/libraries/Zigbee/src/ZigbeeCore.h +++ b/libraries/Zigbee/src/ZigbeeCore.h @@ -11,6 +11,7 @@ #include "aps/esp_zigbee_aps.h" #include #include +#include "ZigbeeTypes.h" #include "ZigbeeEP.h" class ZigbeeEP; @@ -103,6 +104,9 @@ class ZigbeeCore { SemaphoreHandle_t lock; bool _debug; + // Global default response callback + void (*_global_default_response_cb)(zb_cmd_type_t resp_to_cmd, esp_zb_zcl_status_t status, uint8_t endpoint, uint16_t cluster); + bool zigbeeInit(esp_zb_cfg_t *zb_cfg, bool erase_nvs); static void scanCompleteCallback(esp_zb_zdp_status_t zdo_status, uint8_t count, esp_zb_network_descriptor_t *nwk_descriptor); const char *getDeviceTypeString(esp_zb_ha_standard_devices_t deviceId); @@ -176,6 +180,14 @@ class ZigbeeCore { return _debug; } + // Set global default response callback + void onGlobalDefaultResponse(void (*callback)(zb_cmd_type_t resp_to_cmd, esp_zb_zcl_status_t status, uint8_t endpoint, uint16_t cluster)) { + _global_default_response_cb = callback; + } + + // Call global default response callback (for internal use) + void callDefaultResponseCallback(zb_cmd_type_t resp_to_cmd, esp_zb_zcl_status_t status, uint8_t endpoint, uint16_t cluster); + // Friend function declaration to allow access to private members friend void esp_zb_app_signal_handler(esp_zb_app_signal_t *signal_struct); friend bool zb_apsde_data_indication_handler(esp_zb_apsde_data_ind_t ind); diff --git a/libraries/Zigbee/src/ZigbeeEP.cpp b/libraries/Zigbee/src/ZigbeeEP.cpp index efddbdd0368..6b63cae0312 100644 --- a/libraries/Zigbee/src/ZigbeeEP.cpp +++ b/libraries/Zigbee/src/ZigbeeEP.cpp @@ -608,7 +608,17 @@ void ZigbeeEP::removeBoundDevice(zb_device_params_t *device) { log_w("No matching device found for removal"); } -const char *ZigbeeEP::esp_zb_zcl_status_to_name(esp_zb_zcl_status_t status) { +void ZigbeeEP::zbDefaultResponse(const esp_zb_zcl_cmd_default_resp_message_t *message) { + log_v("Default response received for endpoint %d", _endpoint); + log_v("Status code: %s", esp_zb_zcl_status_to_name(message->status_code)); + log_v("Response to command: %d", message->resp_to_cmd); + if (_on_default_response) { + _on_default_response((zb_cmd_type_t)message->resp_to_cmd, message->status_code); + } +} + +// Global function implementation +const char *esp_zb_zcl_status_to_name(esp_zb_zcl_status_t status) { switch (status) { case ESP_ZB_ZCL_STATUS_SUCCESS: return "Success"; case ESP_ZB_ZCL_STATUS_FAIL: return "Fail"; diff --git a/libraries/Zigbee/src/ZigbeeEP.h b/libraries/Zigbee/src/ZigbeeEP.h index a3217cbd066..23217407003 100644 --- a/libraries/Zigbee/src/ZigbeeEP.h +++ b/libraries/Zigbee/src/ZigbeeEP.h @@ -38,6 +38,9 @@ typedef enum { ZB_POWER_SOURCE_BATTERY = 0x03, } zb_power_source_t; +// Global function for converting ZCL status to name +const char *esp_zb_zcl_status_to_name(esp_zb_zcl_status_t status); + /* Zigbee End Device Class */ class ZigbeeEP { public: @@ -138,6 +141,7 @@ class ZigbeeEP { 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 zbDefaultResponse(const esp_zb_zcl_cmd_default_resp_message_t *message); //already implemented virtual void addBoundDevice(zb_device_params_t *device) { _bound_devices.push_back(device); @@ -156,17 +160,21 @@ class ZigbeeEP { _on_identify = callback; } + void onDefaultResponse(void (*callback)(zb_cmd_type_t resp_to_cmd, esp_zb_zcl_status_t status)) { + _on_default_response = callback; + } + + // Convert ZCL status to name + private: char *_read_manufacturer; char *_read_model; void (*_on_identify)(uint16_t time); + void (*_on_default_response)(zb_cmd_type_t resp_to_cmd, esp_zb_zcl_status_t status); time_t _read_time; int32_t _read_timezone; protected: - // Convert ZCL status to name - const char *esp_zb_zcl_status_to_name(esp_zb_zcl_status_t status); - uint8_t _endpoint; esp_zb_ha_standard_devices_t _device_id; esp_zb_endpoint_config_t _ep_config; @@ -179,6 +187,7 @@ class ZigbeeEP { zb_power_source_t _power_source; uint8_t _time_status; + // Friend class declaration to allow access to protected members friend class ZigbeeCore; }; diff --git a/libraries/Zigbee/src/ZigbeeHandlers.cpp b/libraries/Zigbee/src/ZigbeeHandlers.cpp index 5d54e459058..0986056dcd9 100644 --- a/libraries/Zigbee/src/ZigbeeHandlers.cpp +++ b/libraries/Zigbee/src/ZigbeeHandlers.cpp @@ -398,6 +398,16 @@ static esp_err_t zb_cmd_default_resp_handler(const esp_zb_zcl_cmd_default_resp_m "Received default response: from address(0x%x), src_endpoint(%d) to dst_endpoint(%d), cluster(0x%x) with status 0x%x", message->info.src_address.u.short_addr, message->info.src_endpoint, message->info.dst_endpoint, message->info.cluster, message->status_code ); + + // Call global callback if set + Zigbee.callDefaultResponseCallback((zb_cmd_type_t)message->resp_to_cmd, message->status_code, message->info.dst_endpoint, message->info.cluster); + + // 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)->zbDefaultResponse(message); //method zbDefaultResponse is implemented in the common EP class + } + } return ESP_OK; } diff --git a/libraries/Zigbee/src/ZigbeeTypes.h b/libraries/Zigbee/src/ZigbeeTypes.h new file mode 100644 index 00000000000..5025f90db7c --- /dev/null +++ b/libraries/Zigbee/src/ZigbeeTypes.h @@ -0,0 +1,30 @@ +#pragma once + +#include "esp_zigbee_core.h" + +// Foundation Command Types +typedef enum { + ZB_CMD_READ_ATTRIBUTE = 0x00U, /*!< Read attributes command */ + ZB_CMD_READ_ATTRIBUTE_RESPONSE = 0x01U, /*!< Read attributes response command */ + ZB_CMD_WRITE_ATTRIBUTE = 0x02U, /*!< Write attributes foundation command */ + ZB_CMD_WRITE_ATTRIBUTE_UNDIVIDED = 0x03U, /*!< Write attributes undivided command */ + ZB_CMD_WRITE_ATTRIBUTE_RESPONSE = 0x04U, /*!< Write attributes response command */ + ZB_CMD_WRITE_ATTRIBUTE_NO_RESPONSE = 0x05U, /*!< Write attributes no response command */ + ZB_CMD_CONFIGURE_REPORTING = 0x06U, /*!< Configure reporting command */ + ZB_CMD_CONFIGURE_REPORTING_RESPONSE = 0x07U, /*!< Configure reporting response command */ + ZB_CMD_READ_REPORTING_CONFIG = 0x08U, /*!< Read reporting config command */ + ZB_CMD_READ_REPORTING_CONFIG_RESPONSE = 0x09U, /*!< Read reporting config response command */ + ZB_CMD_REPORT_ATTRIBUTE = 0x0aU, /*!< Report attribute command */ + ZB_CMD_DEFAULT_RESPONSE = 0x0bU, /*!< Default response command */ + ZB_CMD_DISCOVER_ATTRIBUTES = 0x0cU, /*!< Discover attributes command */ + ZB_CMD_DISCOVER_ATTRIBUTES_RESPONSE = 0x0dU, /*!< Discover attributes response command */ + ZB_CMD_READ_ATTRIBUTE_STRUCTURED = 0x0eU, /*!< Read attributes structured */ + ZB_CMD_WRITE_ATTRIBUTE_STRUCTURED = 0x0fU, /*!< Write attributes structured */ + ZB_CMD_WRITE_ATTRIBUTE_STRUCTURED_RESPONSE = 0x10U, /*!< Write attributes structured response */ + ZB_CMD_DISCOVER_COMMANDS_RECEIVED = 0x11U, /*!< Discover Commands Received command */ + ZB_CMD_DISCOVER_COMMANDS_RECEIVED_RESPONSE = 0x12U, /*!< Discover Commands Received response command */ + ZB_CMD_DISCOVER_COMMANDS_GENERATED = 0x13U, /*!< Discover Commands Generated command */ + ZB_CMD_DISCOVER_COMMANDS_GENERATED_RESPONSE = 0x14U, /*!< Discover Commands Generated response command */ + ZB_CMD_DISCOVER_ATTRIBUTES_EXTENDED = 0x15U, /*!< Discover attributes extended command */ + ZB_CMD_DISCOVER_ATTRIBUTES_EXTENDED_RESPONSE = 0x16U, /*!< Discover attributes extended response command */ +} zb_cmd_type_t; From 995e603d3af8305959948a46ff0793c72e5c1687 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jan=20Proch=C3=A1zka?= <90197375+P-R-O-C-H-Y@users.noreply.github.com> Date: Mon, 21 Jul 2025 14:47:43 +0200 Subject: [PATCH 132/173] fix(zigbee): Replace assert with error log to solve immediate crash (#11614) * fix(zigbee): Replace assert with error log to solve immediate crash * ci(pre-commit): Apply automatic fixes --------- Co-authored-by: pre-commit-ci-lite[bot] <117423508+pre-commit-ci-lite[bot]@users.noreply.github.com> --- libraries/Zigbee/src/ZigbeeCore.cpp | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/libraries/Zigbee/src/ZigbeeCore.cpp b/libraries/Zigbee/src/ZigbeeCore.cpp index bf652642ac1..90b29cf9d0a 100644 --- a/libraries/Zigbee/src/ZigbeeCore.cpp +++ b/libraries/Zigbee/src/ZigbeeCore.cpp @@ -238,7 +238,9 @@ void ZigbeeCore::closeNetwork() { } static void bdb_start_top_level_commissioning_cb(uint8_t mode_mask) { - ESP_ERROR_CHECK(esp_zb_bdb_start_top_level_commissioning(mode_mask)); + if (esp_zb_bdb_start_top_level_commissioning(mode_mask) != ESP_OK) { + log_e("Failed to start Zigbee commissioning"); + } } void esp_zb_app_signal_handler(esp_zb_app_signal_t *signal_struct) { From f1712943b4960749316612bcf0fb4caf5d74efa0 Mon Sep 17 00:00:00 2001 From: Me No Dev Date: Mon, 21 Jul 2025 15:48:05 +0300 Subject: [PATCH 133/173] fix(ppp): Detach PPP RST pin from periman on end (#11620) --- libraries/PPP/src/PPP.cpp | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/libraries/PPP/src/PPP.cpp b/libraries/PPP/src/PPP.cpp index 87fee6920c3..2a5c5760287 100644 --- a/libraries/PPP/src/PPP.cpp +++ b/libraries/PPP/src/PPP.cpp @@ -432,6 +432,11 @@ void PPPClass::end(void) { _pin_cts = -1; perimanClearPinBus(pin); } + if (_pin_rst != -1) { + pin = _pin_rst; + _pin_rst = -1; + perimanClearPinBus(pin); + } _mode = ESP_MODEM_MODE_COMMAND; } From 4a3c6d7fbbb21ac7439e145680f8201c00b11b67 Mon Sep 17 00:00:00 2001 From: Me No Dev Date: Mon, 21 Jul 2025 15:49:15 +0300 Subject: [PATCH 134/173] feat(netif): Allow setting interface's routing priority (#11617) * feat(netif): Allow setting interface's routing priority * feat(netif): Rename route priority method names and add notes * feat(netif): Print route prio for each interface --- libraries/Network/src/NetworkInterface.cpp | 28 ++++++++++++++++++++-- libraries/Network/src/NetworkInterface.h | 5 +++- 2 files changed, 30 insertions(+), 3 deletions(-) diff --git a/libraries/Network/src/NetworkInterface.cpp b/libraries/Network/src/NetworkInterface.cpp index 01790ec2493..06cf2a377b0 100644 --- a/libraries/Network/src/NetworkInterface.cpp +++ b/libraries/Network/src/NetworkInterface.cpp @@ -606,13 +606,33 @@ int NetworkInterface::impl_index() const { return esp_netif_get_netif_impl_index(_esp_netif); } -int NetworkInterface::route_prio() const { +/** + * Every netif has a parameter named route_prio, you can refer to file esp_netif_defaults.h. + * A higher value of route_prio indicates a higher priority. + * The active interface with highest priority will be used for default route (gateway). + * Defaults are: STA=100, BR=70, ETH=50, PPP=20, AP/NAN=10 + */ +int NetworkInterface::getRoutePrio() const { if (_esp_netif == NULL) { return -1; } return esp_netif_get_route_prio(_esp_netif); } +#if ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 5, 0) +int NetworkInterface::setRoutePrio(int prio) { + if (_esp_netif == NULL) { + return -1; + } + return esp_netif_set_route_prio(_esp_netif, prio); +} +#endif + +/** + * This API overrides the automatic configuration of the default interface based on the route_prio + * If the selected netif is set default using this API, no other interface could be set-default disregarding + * its route_prio number (unless the selected netif gets destroyed) + */ bool NetworkInterface::setDefault() { if (_esp_netif == NULL) { return false; @@ -819,7 +839,11 @@ size_t NetworkInterface::printTo(Print &out) const { if (flags & ESP_NETIF_FLAG_MLDV6_REPORT) { bytes += out.print(",V6_REP"); } - bytes += out.println(")"); + bytes += out.print(")"); + + bytes += out.print(" PRIO: "); + bytes += out.print(getRoutePrio()); + bytes += out.println(""); bytes += out.print(" "); bytes += out.print("ether "); diff --git a/libraries/Network/src/NetworkInterface.h b/libraries/Network/src/NetworkInterface.h index 4f97181d4fd..fd26df77697 100644 --- a/libraries/Network/src/NetworkInterface.h +++ b/libraries/Network/src/NetworkInterface.h @@ -57,7 +57,10 @@ class NetworkInterface : public Printable { const char *desc() const; String impl_name() const; int impl_index() const; - int route_prio() const; + int getRoutePrio() const; +#if ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 5, 0) + int setRoutePrio(int prio); +#endif bool setDefault(); bool isDefault() const; From c369dca062ee99bf8b20ffee89537f3c66d9ea52 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jan=20Proch=C3=A1zka?= <90197375+P-R-O-C-H-Y@users.noreply.github.com> Date: Mon, 21 Jul 2025 23:35:28 +0200 Subject: [PATCH 135/173] feat(docs): Add Zigbee library API documentation (#11525) * feat(docs): Add Zigbee library documentation * fix: Remove helper scripts * fix: Proper class naming for better readability * fix(docs): Fix typos Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> * ci(pre-commit): Apply automatic fixes * fix(docs): Precommit fixes * fix(docs): Precommit fixes --------- Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> Co-authored-by: pre-commit-ci-lite[bot] <117423508+pre-commit-ci-lite[bot]@users.noreply.github.com> --- docs/en/libraries.rst | 12 + docs/en/zigbee/ep_analog.rst | 280 ++++++++++++++ docs/en/zigbee/ep_binary.rst | 137 +++++++ docs/en/zigbee/ep_carbon_dioxide_sensor.rst | 105 ++++++ docs/en/zigbee/ep_color_dimmable_light.rst | 223 +++++++++++ docs/en/zigbee/ep_color_dimmer_switch.rst | 186 +++++++++ docs/en/zigbee/ep_contact_switch.rst | 95 +++++ docs/en/zigbee/ep_dimmable_light.rst | 138 +++++++ docs/en/zigbee/ep_door_window_handle.rst | 95 +++++ docs/en/zigbee/ep_electrical_measurement.rst | 254 +++++++++++++ docs/en/zigbee/ep_flow_sensor.rst | 119 ++++++ docs/en/zigbee/ep_gateway.rst | 35 ++ docs/en/zigbee/ep_illuminance_sensor.rst | 109 ++++++ docs/en/zigbee/ep_light.rst | 88 +++++ docs/en/zigbee/ep_occupancy_sensor.rst | 81 ++++ docs/en/zigbee/ep_pm25_sensor.rst | 109 ++++++ docs/en/zigbee/ep_power_outlet.rst | 88 +++++ docs/en/zigbee/ep_pressure_sensor.rst | 109 ++++++ docs/en/zigbee/ep_range_extender.rst | 34 ++ docs/en/zigbee/ep_switch.rst | 136 +++++++ docs/en/zigbee/ep_temperature_sensor.rst | 184 +++++++++ docs/en/zigbee/ep_thermostat.rst | 238 ++++++++++++ docs/en/zigbee/ep_vibration_sensor.rst | 87 +++++ docs/en/zigbee/ep_wind_speed_sensor.rst | 119 ++++++ docs/en/zigbee/ep_window_covering.rst | 233 ++++++++++++ docs/en/zigbee/zigbee.rst | 159 ++++++++ docs/en/zigbee/zigbee_core.rst | 375 +++++++++++++++++++ docs/en/zigbee/zigbee_ep.rst | 357 ++++++++++++++++++ 28 files changed, 4185 insertions(+) create mode 100644 docs/en/zigbee/ep_analog.rst create mode 100644 docs/en/zigbee/ep_binary.rst create mode 100644 docs/en/zigbee/ep_carbon_dioxide_sensor.rst create mode 100644 docs/en/zigbee/ep_color_dimmable_light.rst create mode 100644 docs/en/zigbee/ep_color_dimmer_switch.rst create mode 100644 docs/en/zigbee/ep_contact_switch.rst create mode 100644 docs/en/zigbee/ep_dimmable_light.rst create mode 100644 docs/en/zigbee/ep_door_window_handle.rst create mode 100644 docs/en/zigbee/ep_electrical_measurement.rst create mode 100644 docs/en/zigbee/ep_flow_sensor.rst create mode 100644 docs/en/zigbee/ep_gateway.rst create mode 100644 docs/en/zigbee/ep_illuminance_sensor.rst create mode 100644 docs/en/zigbee/ep_light.rst create mode 100644 docs/en/zigbee/ep_occupancy_sensor.rst create mode 100644 docs/en/zigbee/ep_pm25_sensor.rst create mode 100644 docs/en/zigbee/ep_power_outlet.rst create mode 100644 docs/en/zigbee/ep_pressure_sensor.rst create mode 100644 docs/en/zigbee/ep_range_extender.rst create mode 100644 docs/en/zigbee/ep_switch.rst create mode 100644 docs/en/zigbee/ep_temperature_sensor.rst create mode 100644 docs/en/zigbee/ep_thermostat.rst create mode 100644 docs/en/zigbee/ep_vibration_sensor.rst create mode 100644 docs/en/zigbee/ep_wind_speed_sensor.rst create mode 100644 docs/en/zigbee/ep_window_covering.rst create mode 100644 docs/en/zigbee/zigbee.rst create mode 100644 docs/en/zigbee/zigbee_core.rst create mode 100644 docs/en/zigbee/zigbee_ep.rst diff --git a/docs/en/libraries.rst b/docs/en/libraries.rst index 525a5c4ba26..07f0978be68 100644 --- a/docs/en/libraries.rst +++ b/docs/en/libraries.rst @@ -91,3 +91,15 @@ The Arduino ESP32 offers some unique APIs, described in this section: :glob: api/* + +Zigbee APIs +----------- + +.. toctree:: + :maxdepth: 1 + :glob: + + zigbee/zigbee + zigbee/zigbee_core + zigbee/zigbee_ep + zigbee/ep_* diff --git a/docs/en/zigbee/ep_analog.rst b/docs/en/zigbee/ep_analog.rst new file mode 100644 index 00000000000..45740007881 --- /dev/null +++ b/docs/en/zigbee/ep_analog.rst @@ -0,0 +1,280 @@ +############ +ZigbeeAnalog +############ + +About +----- + +The ``ZigbeeAnalog`` class provides analog input and output endpoints for Zigbee networks. This endpoint implements the Zigbee Home Automation (HA) standard for analog signal processing and control. +Analog Input (AI) is meant to be used for sensors that provide an analog signal, such as temperature, humidity, pressure to be sent to the coordinator. +Analog Output (AO) is meant to be used for actuators that require an analog signal, such as dimmers, valves, etc. to be controlled by the coordinator. + + +Common API +---------- + +Constructor +*********** + +ZigbeeAnalog +^^^^^^^^^^^^ + +Creates a new Zigbee analog endpoint. + +.. code-block:: arduino + + ZigbeeAnalog(uint8_t endpoint); + +* ``endpoint`` - Endpoint number (1-254) + +Cluster Management +****************** + +addAnalogInput +^^^^^^^^^^^^^^ + +Adds analog input cluster to the endpoint. + +.. code-block:: arduino + + bool addAnalogInput(); + +This function will return ``true`` if successful, ``false`` otherwise. + +addAnalogOutput +^^^^^^^^^^^^^^^ + +Adds analog output cluster to the endpoint. + +.. code-block:: arduino + + bool addAnalogOutput(); + +This function will return ``true`` if successful, ``false`` otherwise. + +Analog Input API +---------------- + +Configuration Methods +********************* + +setAnalogInputApplication +^^^^^^^^^^^^^^^^^^^^^^^^^ + +Sets the application type for the analog input. + +.. code-block:: arduino + + bool setAnalogInputApplication(uint32_t application_type); + +* ``application_type`` - Application type constant (see esp_zigbee_zcl_analog_input.h for values) + +This function will return ``true`` if successful, ``false`` otherwise. + +setAnalogInputDescription +^^^^^^^^^^^^^^^^^^^^^^^^^ + +Sets a custom description for the analog input. + +.. code-block:: arduino + + bool setAnalogInputDescription(const char *description); + +* ``description`` - Description string + +This function will return ``true`` if successful, ``false`` otherwise. + +setAnalogInputResolution +^^^^^^^^^^^^^^^^^^^^^^^^ + +Sets the resolution for the analog input. + +.. code-block:: arduino + + bool setAnalogInputResolution(float resolution); + +* ``resolution`` - Resolution value + +This function will return ``true`` if successful, ``false`` otherwise. + +setAnalogInputMinMax +^^^^^^^^^^^^^^^^^^^^ + +Sets the minimum and maximum values for the analog input. + +.. code-block:: arduino + + bool setAnalogInputMinMax(float min, float max); + +* ``min`` - Minimum value +* ``max`` - Maximum value + +This function will return ``true`` if successful, ``false`` otherwise. + +Value Control +************* + +setAnalogInput +^^^^^^^^^^^^^^ + +Sets the analog input value. + +.. code-block:: arduino + + bool setAnalogInput(float analog); + +* ``analog`` - Analog input value + +This function will return ``true`` if successful, ``false`` otherwise. + +Reporting Methods +***************** + +setAnalogInputReporting +^^^^^^^^^^^^^^^^^^^^^^^ + +Sets the reporting configuration for analog input. + +.. code-block:: arduino + + bool setAnalogInputReporting(uint16_t min_interval, uint16_t max_interval, float delta); + +* ``min_interval`` - Minimum reporting interval in seconds +* ``max_interval`` - Maximum reporting interval in seconds +* ``delta`` - Minimum change in value to trigger a report + +This function will return ``true`` if successful, ``false`` otherwise. + +reportAnalogInput +^^^^^^^^^^^^^^^^^ + +Manually reports the current analog input value. + +.. code-block:: arduino + + bool reportAnalogInput(); + +This function will return ``true`` if successful, ``false`` otherwise. + +Analog Output API +----------------- + +Configuration Methods +********************* + +setAnalogOutputApplication +^^^^^^^^^^^^^^^^^^^^^^^^^^ + +Sets the application type for the analog output. + +.. code-block:: arduino + + bool setAnalogOutputApplication(uint32_t application_type); + +* ``application_type`` - Application type constant (see esp_zigbee_zcl_analog_output.h for values) + +This function will return ``true`` if successful, ``false`` otherwise. + +setAnalogOutputDescription +^^^^^^^^^^^^^^^^^^^^^^^^^^ + +Sets a custom description for the analog output. + +.. code-block:: arduino + + bool setAnalogOutputDescription(const char *description); + +* ``description`` - Description string + +This function will return ``true`` if successful, ``false`` otherwise. + +setAnalogOutputResolution +^^^^^^^^^^^^^^^^^^^^^^^^^ + +Sets the resolution for the analog output. + +.. code-block:: arduino + + bool setAnalogOutputResolution(float resolution); + +* ``resolution`` - Resolution value + +This function will return ``true`` if successful, ``false`` otherwise. + +setAnalogOutputMinMax +^^^^^^^^^^^^^^^^^^^^^ + +Sets the minimum and maximum values for the analog output. + +.. code-block:: arduino + + bool setAnalogOutputMinMax(float min, float max); + +* ``min`` - Minimum value +* ``max`` - Maximum value + +This function will return ``true`` if successful, ``false`` otherwise. + +Value Control +************* + +setAnalogOutput +^^^^^^^^^^^^^^^ + +Sets the analog output value. + +.. code-block:: arduino + + bool setAnalogOutput(float analog); + +* ``analog`` - Analog output value + +This function will return ``true`` if successful, ``false`` otherwise. + +getAnalogOutput +^^^^^^^^^^^^^^^ + +Gets the current analog output value. + +.. code-block:: arduino + + float getAnalogOutput(); + +This function will return current analog output value. + +Reporting Methods +***************** + +reportAnalogOutput +^^^^^^^^^^^^^^^^^^ + +Manually reports the current analog output value. + +.. code-block:: arduino + + bool reportAnalogOutput(); + +This function will return ``true`` if successful, ``false`` otherwise. + +Event Handling +************** + +onAnalogOutputChange +^^^^^^^^^^^^^^^^^^^^ + +Sets a callback function to be called when the analog output value changes. + +.. code-block:: arduino + + void onAnalogOutputChange(void (*callback)(float analog)); + +* ``callback`` - Function to call when analog output changes + +Example +------- + +Analog Input/Output +******************* + +.. literalinclude:: ../../../libraries/Zigbee/examples/Zigbee_Analog_Input_Output/Zigbee_Analog_Input_Output.ino + :language: arduino diff --git a/docs/en/zigbee/ep_binary.rst b/docs/en/zigbee/ep_binary.rst new file mode 100644 index 00000000000..950e20ef42b --- /dev/null +++ b/docs/en/zigbee/ep_binary.rst @@ -0,0 +1,137 @@ +############ +ZigbeeBinary +############ + +About +----- + +The ``ZigbeeBinary`` class provides an endpoint for binary input/output sensors in Zigbee networks. This endpoint implements the Zigbee Home Automation (HA) standard for binary sensors, supporting various application types for HVAC, security, and general binary sensing. +Binary Input (BI) is meant to be used for sensors that provide a binary signal, such as door/window sensors, motion detectors, etc. to be sent to the network. + +.. note:: + + Binary Output (BO) is not supported yet. + +API Reference +------------- + +Constructor +*********** + +ZigbeeBinary +^^^^^^^^^^^^ + +Creates a new Zigbee binary sensor endpoint. + +.. code-block:: arduino + + ZigbeeBinary(uint8_t endpoint); + +* ``endpoint`` - Endpoint number (1-254) + +Binary Input Application Types +****************************** + +HVAC Application Types +^^^^^^^^^^^^^^^^^^^^^^ + +.. code-block:: arduino + + #define BINARY_INPUT_APPLICATION_TYPE_HVAC_BOILER_STATUS 0x00000003 + #define BINARY_INPUT_APPLICATION_TYPE_HVAC_CHILLER_STATUS 0x00000013 + #define BINARY_INPUT_APPLICATION_TYPE_HVAC_OCCUPANCY 0x00000031 + #define BINARY_INPUT_APPLICATION_TYPE_HVAC_FAN_STATUS 0x00000035 + #define BINARY_INPUT_APPLICATION_TYPE_HVAC_FILTER_STATUS 0x00000036 + #define BINARY_INPUT_APPLICATION_TYPE_HVAC_HEATING_ALARM 0x0000003E + #define BINARY_INPUT_APPLICATION_TYPE_HVAC_COOLING_ALARM 0x0000001D + #define BINARY_INPUT_APPLICATION_TYPE_HVAC_UNIT_ENABLE 0x00000090 + #define BINARY_INPUT_APPLICATION_TYPE_HVAC_OTHER 0x0000FFFF + +Security Application Types +^^^^^^^^^^^^^^^^^^^^^^^^^^ + +.. code-block:: arduino + + #define BINARY_INPUT_APPLICATION_TYPE_SECURITY_GLASS_BREAKAGE_DETECTION_0 0x01000000 + #define BINARY_INPUT_APPLICATION_TYPE_SECURITY_INTRUSION_DETECTION 0x01000001 + #define BINARY_INPUT_APPLICATION_TYPE_SECURITY_MOTION_DETECTION 0x01000002 + #define BINARY_INPUT_APPLICATION_TYPE_SECURITY_GLASS_BREAKAGE_DETECTION_1 0x01000003 + #define BINARY_INPUT_APPLICATION_TYPE_SECURITY_ZONE_ARMED 0x01000004 + #define BINARY_INPUT_APPLICATION_TYPE_SECURITY_GLASS_BREAKAGE_DETECTION_2 0x01000005 + #define BINARY_INPUT_APPLICATION_TYPE_SECURITY_SMOKE_DETECTION 0x01000006 + #define BINARY_INPUT_APPLICATION_TYPE_SECURITY_CARBON_DIOXIDE_DETECTION 0x01000007 + #define BINARY_INPUT_APPLICATION_TYPE_SECURITY_HEAT_DETECTION 0x01000008 + #define BINARY_INPUT_APPLICATION_TYPE_SECURITY_OTHER 0x0100FFFF + +API Methods +*********** + +addBinaryInput +^^^^^^^^^^^^^^ + +Adds a binary input cluster to the endpoint. + +.. code-block:: arduino + + bool addBinaryInput(); + +This function will return ``true`` if successful, ``false`` otherwise. + +setBinaryInputApplication +^^^^^^^^^^^^^^^^^^^^^^^^^ + +Sets the application type for the binary input. + +.. code-block:: arduino + + bool setBinaryInputApplication(uint32_t application_type); + +* ``application_type`` - Application type constant (see above) + +This function will return ``true`` if successful, ``false`` otherwise. + +setBinaryInputDescription +^^^^^^^^^^^^^^^^^^^^^^^^^ + +Sets a custom description for the binary input. + +.. code-block:: arduino + + bool setBinaryInputDescription(const char *description); + +* ``description`` - Description string + +This function will return ``true`` if successful, ``false`` otherwise. + +setBinaryInput +^^^^^^^^^^^^^^ + +Sets the binary input value. + +.. code-block:: arduino + + bool setBinaryInput(bool input); + +* ``input`` - Binary value (true/false) + +This function will return ``true`` if successful, ``false`` otherwise. + +reportBinaryInput +^^^^^^^^^^^^^^^^^ + +Manually reports the current binary input value. + +.. code-block:: arduino + + bool reportBinaryInput(); + +This function will return ``true`` if successful, ``false`` otherwise. + +Example +------- + +Binary Input Implementation +**************************** + +.. literalinclude:: ../../../libraries/Zigbee/examples/Zigbee_Binary_Input/Zigbee_Binary_Input.ino + :language: arduino diff --git a/docs/en/zigbee/ep_carbon_dioxide_sensor.rst b/docs/en/zigbee/ep_carbon_dioxide_sensor.rst new file mode 100644 index 00000000000..6f219a16fd0 --- /dev/null +++ b/docs/en/zigbee/ep_carbon_dioxide_sensor.rst @@ -0,0 +1,105 @@ +######################### +ZigbeeCarbonDioxideSensor +######################### + +About +----- + +The ``ZigbeeCarbonDioxideSensor`` class provides a CO2 sensor endpoint for Zigbee networks. This endpoint implements the Zigbee Home Automation (HA) standard for carbon dioxide measurement devices. + +API Reference +------------- + +Constructor +*********** + +ZigbeeCarbonDioxideSensor +^^^^^^^^^^^^^^^^^^^^^^^^^ + +Creates a new Zigbee CO2 sensor endpoint. + +.. code-block:: arduino + + ZigbeeCarbonDioxideSensor(uint8_t endpoint); + +* ``endpoint`` - Endpoint number (1-254) + +API Methods +*********** + +setCarbonDioxide +^^^^^^^^^^^^^^^^ + +Sets the CO2 concentration measurement value. + +.. code-block:: arduino + + bool setCarbonDioxide(float carbon_dioxide); + +* ``carbon_dioxide`` - CO2 concentration value in ppm + +This function will return ``true`` if successful, ``false`` otherwise. + +setMinMaxValue +^^^^^^^^^^^^^^ + +Sets the minimum and maximum measurement values. + +.. code-block:: arduino + + bool setMinMaxValue(float min, float max); + +* ``min`` - Minimum CO2 concentration value in ppm +* ``max`` - Maximum CO2 concentration value in ppm + +This function will return ``true`` if successful, ``false`` otherwise. + +setTolerance +^^^^^^^^^^^^ + +Sets the tolerance value for measurements. + +.. code-block:: arduino + + bool setTolerance(float tolerance); + +* ``tolerance`` - Tolerance value in ppm + +This function will return ``true`` if successful, ``false`` otherwise. + +setReporting +^^^^^^^^^^^^ + +Sets the reporting configuration for CO2 measurements. + +.. code-block:: arduino + + bool setReporting(uint16_t min_interval, uint16_t max_interval, uint16_t delta); + +* ``min_interval`` - Minimum reporting interval in seconds +* ``max_interval`` - Maximum reporting interval in seconds +* ``delta`` - Minimum change required to trigger a report in ppm + +**Note:** Delta reporting is currently not supported by the carbon dioxide sensor. + +This function will return ``true`` if successful, ``false`` otherwise. + +report +^^^^^^ + +Manually reports the current CO2 concentration value. + +.. code-block:: arduino + + bool report(); + +This function will return ``true`` if successful, ``false`` otherwise. + +Example +------- + +CO2 Sensor Implementation +************************* + +.. literalinclude:: ../../../libraries/Zigbee/examples/Zigbee_CarbonDioxide_Sensor/Zigbee_CarbonDioxide_Sensor.ino + :language: arduino diff --git a/docs/en/zigbee/ep_color_dimmable_light.rst b/docs/en/zigbee/ep_color_dimmable_light.rst new file mode 100644 index 00000000000..b2fa6d0bf91 --- /dev/null +++ b/docs/en/zigbee/ep_color_dimmable_light.rst @@ -0,0 +1,223 @@ +######################## +ZigbeeColorDimmableLight +######################## + +About +----- + +The ``ZigbeeColorDimmableLight`` class provides an endpoint for color dimmable lights in Zigbee networks. This endpoint implements the Zigbee Home Automation (HA) standard for color lighting devices, supporting RGB color control, dimming, and scene management. + +**Features:** +* On/off control +* Brightness level control (0-100%) +* RGB color control +* HSV color support +* Scene and group support +* Automatic state restoration +* Integration with common endpoint features (binding, OTA, etc.) +* Zigbee HA standard compliance + +**Use Cases:** +* Smart RGB light bulbs +* Color-changing LED strips +* Mood lighting systems +* Entertainment lighting +* Architectural lighting +* Smart home color lighting + +API Reference +------------- + +Constructor +*********** + +ZigbeeColorDimmableLight +^^^^^^^^^^^^^^^^^^^^^^^^ + +Creates a new Zigbee color dimmable light endpoint. + +.. code-block:: arduino + + ZigbeeColorDimmableLight(uint8_t endpoint); + +* ``endpoint`` - Endpoint number (1-254) + +Callback Functions +****************** + +onLightChange +^^^^^^^^^^^^^ + +Sets the callback function for light state changes. + +.. code-block:: arduino + + void onLightChange(void (*callback)(bool, uint8_t, uint8_t, uint8_t, uint8_t)); + +* ``callback`` - Function pointer to the light change callback (state, red, green, blue, level) + +Control Methods +*************** + +setLightState +^^^^^^^^^^^^^ + +Sets the light on/off state. + +.. code-block:: arduino + + bool setLightState(bool state); + +* ``state`` - Light state (true = on, false = off) + +This function will return ``true`` if successful, ``false`` otherwise. + +setLightLevel +^^^^^^^^^^^^^ + +Sets the light brightness level. + +.. code-block:: arduino + + bool setLightLevel(uint8_t level); + +* ``level`` - Brightness level (0-100, where 0 is off, 100 is full brightness) + +This function will return ``true`` if successful, ``false`` otherwise. + +setLightColor (RGB) +^^^^^^^^^^^^^^^^^^^ + +Sets the light color using RGB values. + +.. code-block:: arduino + + bool setLightColor(uint8_t red, uint8_t green, uint8_t blue); + bool setLightColor(espRgbColor_t rgb_color); + +* ``red`` - Red component (0-255) +* ``green`` - Green component (0-255) +* ``blue`` - Blue component (0-255) +* ``rgb_color`` - RGB color structure + +This function will return ``true`` if successful, ``false`` otherwise. + +setLightColor (HSV) +^^^^^^^^^^^^^^^^^^^ + +Sets the light color using HSV values. + +.. code-block:: arduino + + bool setLightColor(espHsvColor_t hsv_color); + +* ``hsv_color`` - HSV color structure + +This function will return ``true`` if successful, ``false`` otherwise. + +setLight +^^^^^^^^ + +Sets all light parameters at once. + +.. code-block:: arduino + + bool setLight(bool state, uint8_t level, uint8_t red, uint8_t green, uint8_t blue); + +* ``state`` - Light state (true/false) +* ``level`` - Brightness level (0-100) +* ``red`` - Red component (0-255) +* ``green`` - Green component (0-255) +* ``blue`` - Blue component (0-255) + +This function will return ``true`` if successful, ``false`` otherwise. + +State Retrieval Methods +*********************** + +getLightState +^^^^^^^^^^^^^ + +Gets the current light state. + +.. code-block:: arduino + + bool getLightState(); + +This function will return current light state (true = on, false = off). + +getLightLevel +^^^^^^^^^^^^^ + +Gets the current brightness level. + +.. code-block:: arduino + + uint8_t getLightLevel(); + +This function will return current brightness level (0-100). + +getLightColor +^^^^^^^^^^^^^ + +Gets the current RGB color. + +.. code-block:: arduino + + espRgbColor_t getLightColor(); + +This function will return current RGB color structure. + +getLightRed +^^^^^^^^^^^ + +Gets the current red component. + +.. code-block:: arduino + + uint8_t getLightRed(); + +This function will return current red component (0-255). + +getLightGreen +^^^^^^^^^^^^^ + +Gets the current green component. + +.. code-block:: arduino + + uint8_t getLightGreen(); + +This function will return current green component (0-255). + +getLightBlue +^^^^^^^^^^^^ + +Gets the current blue component. + +.. code-block:: arduino + + uint8_t getLightBlue(); + +This function will return current blue component (0-255). + +Utility Methods +*************** + +restoreLight +^^^^^^^^^^^^ + +Restores the light to its last known state. + +.. code-block:: arduino + + void restoreLight(); + +Example +------- + +Color Dimmable Light Implementation +*********************************** + +.. literalinclude:: ../../../libraries/Zigbee/examples/Zigbee_Color_Dimmable_Light/Zigbee_Color_Dimmable_Light.ino + :language: arduino diff --git a/docs/en/zigbee/ep_color_dimmer_switch.rst b/docs/en/zigbee/ep_color_dimmer_switch.rst new file mode 100644 index 00000000000..720c5d8f5bf --- /dev/null +++ b/docs/en/zigbee/ep_color_dimmer_switch.rst @@ -0,0 +1,186 @@ +####################### +ZigbeeColorDimmerSwitch +####################### + +About +----- + +The ``ZigbeeColorDimmerSwitch`` class provides a color dimmer switch endpoint for Zigbee networks. This endpoint implements the Zigbee Home Automation (HA) standard for advanced lighting control switches that can control both dimming and color of lights. + +**Features:** +* On/off control for bound lights +* Brightness level control (0-100%) +* Color control (RGB, HSV) +* Color temperature control +* Scene and group support +* Special effects and timed operations +* Integration with common endpoint features (binding, OTA, etc.) +* Zigbee HA standard compliance + +**Use Cases:** +* Smart lighting switches +* Color control remotes +* Advanced lighting controllers +* Smart home lighting automation +* Entertainment lighting control + +API Reference +------------- + +Constructor +*********** + +ZigbeeColorDimmerSwitch +^^^^^^^^^^^^^^^^^^^^^^^ + +Creates a new Zigbee color dimmer switch endpoint. + +.. code-block:: arduino + + ZigbeeColorDimmerSwitch(uint8_t endpoint); + +* ``endpoint`` - Endpoint number (1-254) + +Basic Control Commands +********************** + +lightToggle +^^^^^^^^^^^ + +Toggles the state of bound lights (on to off, or off to on). + +.. code-block:: arduino + + void lightToggle(); + void lightToggle(uint16_t group_addr); + void lightToggle(uint8_t endpoint, uint16_t short_addr); + void lightToggle(uint8_t endpoint, esp_zb_ieee_addr_t ieee_addr); + +* ``group_addr`` - Group address to control (optional) +* ``endpoint`` - Target device endpoint (optional) +* ``short_addr`` - Target device short address (optional) +* ``ieee_addr`` - Target device IEEE address (optional) + +lightOn +^^^^^^^ + +Turns on bound lights. + +.. code-block:: arduino + + void lightOn(); + void lightOn(uint16_t group_addr); + void lightOn(uint8_t endpoint, uint16_t short_addr); + void lightOn(uint8_t endpoint, esp_zb_ieee_addr_t ieee_addr); + +* ``group_addr`` - Group address to control (optional) +* ``endpoint`` - Target device endpoint (optional) +* ``short_addr`` - Target device short address (optional) +* ``ieee_addr`` - Target device IEEE address (optional) + +lightOff +^^^^^^^^ + +Turns off bound lights. + +.. code-block:: arduino + + void lightOff(); + void lightOff(uint16_t group_addr); + void lightOff(uint8_t endpoint, uint16_t short_addr); + void lightOff(uint8_t endpoint, esp_zb_ieee_addr_t ieee_addr); + +* ``group_addr`` - Group address to control (optional) +* ``endpoint`` - Target device endpoint (optional) +* ``short_addr`` - Target device short address (optional) +* ``ieee_addr`` - Target device IEEE address (optional) + +Dimmer Control Commands +*********************** + +setLightLevel +^^^^^^^^^^^^^ + +Sets the brightness level of bound lights. + +.. code-block:: arduino + + void setLightLevel(uint8_t level); + void setLightLevel(uint8_t level, uint16_t group_addr); + void setLightLevel(uint8_t level, uint8_t endpoint, uint16_t short_addr); + void setLightLevel(uint8_t level, uint8_t endpoint, esp_zb_ieee_addr_t ieee_addr); + +* ``level`` - Brightness level (0-100, where 0 is off, 100 is full brightness) +* ``group_addr`` - Group address to control (optional) +* ``endpoint`` - Target device endpoint (optional) +* ``short_addr`` - Target device short address (optional) +* ``ieee_addr`` - Target device IEEE address (optional) + +Color Control Commands +********************** + +setLightColor +^^^^^^^^^^^^^ + +Sets the color of bound lights using RGB values. + +.. code-block:: arduino + + void setLightColor(uint8_t red, uint8_t green, uint8_t blue); + void setLightColor(uint8_t red, uint8_t green, uint8_t blue, uint16_t group_addr); + void setLightColor(uint8_t red, uint8_t green, uint8_t blue, uint8_t endpoint, uint16_t short_addr); + void setLightColor(uint8_t red, uint8_t green, uint8_t blue, uint8_t endpoint, esp_zb_ieee_addr_t ieee_addr); + +* ``red`` - Red component (0-255) +* ``green`` - Green component (0-255) +* ``blue`` - Blue component (0-255) +* ``group_addr`` - Group address to control (optional) +* ``endpoint`` - Target device endpoint (optional) +* ``short_addr`` - Target device short address (optional) +* ``ieee_addr`` - Target device IEEE address (optional) + +Advanced Control Commands +************************* + +lightOffWithEffect +^^^^^^^^^^^^^^^^^^ + +Turns off lights with a specific effect. + +.. code-block:: arduino + + void lightOffWithEffect(uint8_t effect_id, uint8_t effect_variant); + +* ``effect_id`` - Effect identifier +* ``effect_variant`` - Effect variant + +lightOnWithTimedOff +^^^^^^^^^^^^^^^^^^^ + +Turns on lights with automatic turn-off after specified time. + +.. code-block:: arduino + + void lightOnWithTimedOff(uint8_t on_off_control, uint16_t time_on, uint16_t time_off); + +* ``on_off_control`` - Control byte +* ``time_on`` - Time to stay on (in 1/10th seconds) +* ``time_off`` - Time to stay off (in 1/10th seconds) + +lightOnWithSceneRecall +^^^^^^^^^^^^^^^^^^^^^^ + +Turns on lights with scene recall. + +.. code-block:: arduino + + void lightOnWithSceneRecall(); + +Example +------- + +Color Dimmer Switch Implementation +********************************** + +.. literalinclude:: ../../../libraries/Zigbee/examples/Zigbee_Color_Dimmer_Switch/Zigbee_Color_Dimmer_Switch.ino + :language: arduino diff --git a/docs/en/zigbee/ep_contact_switch.rst b/docs/en/zigbee/ep_contact_switch.rst new file mode 100644 index 00000000000..f7f6dc15c66 --- /dev/null +++ b/docs/en/zigbee/ep_contact_switch.rst @@ -0,0 +1,95 @@ +################### +ZigbeeContactSwitch +################### + +About +----- + +The ``ZigbeeContactSwitch`` class provides a contact switch endpoint for Zigbee networks. This endpoint implements the Zigbee Home Automation (HA) standard for door/window contact sensors and other binary contact devices. + +**Features:** +* Contact state detection (open/closed) +* Configurable application types +* Automatic reporting capabilities +* Integration with common endpoint features (binding, OTA, etc.) +* Zigbee HA standard compliance + +**Use Cases:** +* Door and window sensors +* Security system contacts +* Cabinet and drawer sensors +* Industrial contact monitoring +* Smart home security applications + +API Reference +------------- + +Constructor +*********** + +ZigbeeContactSwitch +^^^^^^^^^^^^^^^^^^^ + +Creates a new Zigbee contact switch endpoint. + +.. code-block:: arduino + + ZigbeeContactSwitch(uint8_t endpoint); + +* ``endpoint`` - Endpoint number (1-254) + +API Methods +*********** + +setClosed +^^^^^^^^^ + +Sets the contact switch to closed state. + +.. code-block:: arduino + + bool setClosed(); + +This function will return ``true`` if successful, ``false`` otherwise. + +setOpen +^^^^^^^ + +Sets the contact switch to open state. + +.. code-block:: arduino + + bool setOpen(); + +This function will return ``true`` if successful, ``false`` otherwise. + +setIASClientEndpoint +^^^^^^^^^^^^^^^^^^^^ + +Sets the IAS Client endpoint number (default is 1). + +.. code-block:: arduino + + void setIASClientEndpoint(uint8_t ep_number); + +* ``ep_number`` - IAS Client endpoint number + +report +^^^^^^ + +Manually reports the current contact state. + +.. code-block:: arduino + + bool report(); + +This function will return ``true`` if successful, ``false`` otherwise. + +Example +------- + +Contact Switch Implementation +***************************** + +.. literalinclude:: ../../../libraries/Zigbee/examples/Zigbee_Contact_Switch/Zigbee_Contact_Switch.ino + :language: arduino diff --git a/docs/en/zigbee/ep_dimmable_light.rst b/docs/en/zigbee/ep_dimmable_light.rst new file mode 100644 index 00000000000..dcfe6e1fc9d --- /dev/null +++ b/docs/en/zigbee/ep_dimmable_light.rst @@ -0,0 +1,138 @@ +################### +ZigbeeDimmableLight +################### + +About +----- + +The ``ZigbeeDimmableLight`` class provides a dimmable light endpoint for Zigbee networks. This endpoint implements the Zigbee Home Automation (HA) standard for dimmable lighting control with both on/off and brightness level control. + +**Features:** +* On/off control +* Brightness level control (0-100%) +* State and level change callbacks +* Integration with common endpoint features (binding, OTA, etc.) +* Zigbee HA standard compliance + +**Use Cases:** +* Dimmable smart bulbs +* LED strips with brightness control +* Any device requiring both on/off and dimming functionality + +API Reference +------------- + +Constructor +*********** + +ZigbeeDimmableLight +^^^^^^^^^^^^^^^^^^^ + +Creates a new Zigbee dimmable light endpoint. + +.. code-block:: arduino + + ZigbeeDimmableLight(uint8_t endpoint); + +* ``endpoint`` - Endpoint number (1-254) + +Light Control +************* + +setLightState +^^^^^^^^^^^^^ + +Sets only the light state (on or off) without changing the brightness level. + +.. code-block:: arduino + + bool setLightState(bool state); + +* ``state`` - ``true`` to turn on, ``false`` to turn off + +This function will return ``true`` if successful, ``false`` otherwise. + +setLightLevel +^^^^^^^^^^^^^ + +Sets only the brightness level (0-100) without changing the on/off state. + +.. code-block:: arduino + + bool setLightLevel(uint8_t level); + +* ``level`` - Brightness level (0-100, where 0 is off, 100 is full brightness) + +This function will return ``true`` if successful, ``false`` otherwise. + +setLight +^^^^^^^^ + +Sets both the light state and brightness level simultaneously. + +.. code-block:: arduino + + bool setLight(bool state, uint8_t level); + +* ``state`` - ``true`` to turn on, ``false`` to turn off +* ``level`` - Brightness level (0-100) + +This function will return ``true`` if successful, ``false`` otherwise. + +getLightState +^^^^^^^^^^^^^ + +Gets the current light state. + +.. code-block:: arduino + + bool getLightState(); + +This function will return current light state (``true`` = on, ``false`` = off). + +getLightLevel +^^^^^^^^^^^^^ + +Gets the current brightness level. + +.. code-block:: arduino + + uint8_t getLightLevel(); + +This function will return current brightness level (0-100). + +restoreLight +^^^^^^^^^^^^ + +Restores the light state and triggers any registered callbacks. + +.. code-block:: arduino + + void restoreLight(); + +Event Handling +************** + +onLightChange +^^^^^^^^^^^^^ + +Sets a callback function to be called when the light state or level changes. + +.. code-block:: arduino + + void onLightChange(void (*callback)(bool, uint8_t)); + +* ``callback`` - Function to call when light state or level changes + +**Callback Parameters:** +* ``bool state`` - New light state (true = on, false = off) +* ``uint8_t level`` - New brightness level (0-100) + +Example +------- + +Dimmable Light Implementation +***************************** + +.. literalinclude:: ../../../libraries/Zigbee/examples/Zigbee_Dimmable_Light/Zigbee_Dimmable_Light.ino + :language: arduino diff --git a/docs/en/zigbee/ep_door_window_handle.rst b/docs/en/zigbee/ep_door_window_handle.rst new file mode 100644 index 00000000000..53203f463dd --- /dev/null +++ b/docs/en/zigbee/ep_door_window_handle.rst @@ -0,0 +1,95 @@ +###################### +ZigbeeDoorWindowHandle +###################### + +About +----- + +The ``ZigbeeDoorWindowHandle`` class provides a door/window handle endpoint for Zigbee networks. This endpoint implements the Zigbee Home Automation (HA) standard for handle position sensors and other handle-related devices. + +**Features:** +* Handle position detection +* Multiple position states support +* Configurable application types +* Automatic reporting capabilities + + +API Reference +------------- + +Constructor +*********** + +ZigbeeDoorWindowHandle +^^^^^^^^^^^^^^^^^^^^^^ + +Creates a new Zigbee door/window handle endpoint. + +.. code-block:: arduino + + ZigbeeDoorWindowHandle(uint8_t endpoint); + +* ``endpoint`` - Endpoint number (1-254) + +API Methods +*********** + +setClosed +^^^^^^^^^ + +Sets the door/window handle to closed position. + +.. code-block:: arduino + + bool setClosed(); + +This function will return ``true`` if successful, ``false`` otherwise. + +setOpen +^^^^^^^ + +Sets the door/window handle to open position. + +.. code-block:: arduino + + bool setOpen(); + +This function will return ``true`` if successful, ``false`` otherwise. + +setTilted +^^^^^^^^^ + +Sets the door/window handle to tilted position. + +.. code-block:: arduino + + bool setTilted(); + +This function will return ``true`` if successful, ``false`` otherwise. + +setIASClientEndpoint +^^^^^^^^^^^^^^^^^^^^ + +Sets the IAS Client endpoint number (default is 1). + +.. code-block:: arduino + + void setIASClientEndpoint(uint8_t ep_number); + +* ``ep_number`` - IAS Client endpoint number + +report +^^^^^^ + +Manually reports the current handle position. + +.. code-block:: arduino + + bool report(); + +This function will return ``true`` if successful, ``false`` otherwise. + +Example +------- + +*To be added* diff --git a/docs/en/zigbee/ep_electrical_measurement.rst b/docs/en/zigbee/ep_electrical_measurement.rst new file mode 100644 index 00000000000..86410768484 --- /dev/null +++ b/docs/en/zigbee/ep_electrical_measurement.rst @@ -0,0 +1,254 @@ +########################### +ZigbeeElectricalMeasurement +########################### + +About +----- + +The ``ZigbeeElectricalMeasurement`` class provides an endpoint for electrical measurement devices in Zigbee networks. This endpoint implements the Zigbee Home Automation (HA) standard for power monitoring and electrical measurement. + +**Features:** +* AC and DC electrical measurements +* Voltage, current, and power monitoring +* Power factor measurement +* Multi-phase support +* Configurable measurement ranges +* Automatic reporting capabilities +* Integration with common endpoint features (binding, OTA, etc.) +* Zigbee HA standard compliance + +**Use Cases:** +* Smart power monitoring +* Energy monitoring systems +* Solar panel monitoring +* Battery monitoring systems +* Electrical load monitoring + +Common API +---------- + +Constructor +*********** + +ZigbeeElectricalMeasurement +^^^^^^^^^^^^^^^^^^^^^^^^^^^ + +Creates a new Zigbee electrical measurement endpoint. + +.. code-block:: arduino + + ZigbeeElectricalMeasurement(uint8_t endpoint); + +* ``endpoint`` - Endpoint number (1-254) + +DC Electrical Measurement +************************* + +addDCMeasurement +^^^^^^^^^^^^^^^^ + +Adds a DC measurement type to the endpoint. + +.. code-block:: arduino + + bool addDCMeasurement(ZIGBEE_DC_MEASUREMENT_TYPE measurement_type); + +* ``measurement_type`` - DC measurement type constant (ZIGBEE_DC_MEASUREMENT_TYPE_VOLTAGE, ZIGBEE_DC_MEASUREMENT_TYPE_CURRENT, ZIGBEE_DC_MEASUREMENT_TYPE_POWER) + +This function will return ``true`` if successful, ``false`` otherwise. + +setDCMeasurement +^^^^^^^^^^^^^^^^ + +Sets the DC measurement value for a specific measurement type. + +.. code-block:: arduino + + bool setDCMeasurement(ZIGBEE_DC_MEASUREMENT_TYPE measurement_type, int16_t value); + +* ``measurement_type`` - DC measurement type constant (ZIGBEE_DC_MEASUREMENT_TYPE_VOLTAGE, ZIGBEE_DC_MEASUREMENT_TYPE_CURRENT, ZIGBEE_DC_MEASUREMENT_TYPE_POWER) +* ``value`` - Measurement value + +This function will return ``true`` if successful, ``false`` otherwise. + +setDCMinMaxValue +^^^^^^^^^^^^^^^^ + +Sets the minimum and maximum values for a DC measurement type. + +.. code-block:: arduino + + bool setDCMinMaxValue(ZIGBEE_DC_MEASUREMENT_TYPE measurement_type, int16_t min, int16_t max); + +* ``measurement_type`` - DC measurement type constant (ZIGBEE_DC_MEASUREMENT_TYPE_VOLTAGE, ZIGBEE_DC_MEASUREMENT_TYPE_CURRENT, ZIGBEE_DC_MEASUREMENT_TYPE_POWER) +* ``min`` - Minimum value +* ``max`` - Maximum value + +This function will return ``true`` if successful, ``false`` otherwise. + +setDCMultiplierDivisor +^^^^^^^^^^^^^^^^^^^^^^ + +Sets the multiplier and divisor for scaling DC measurements. + +.. code-block:: arduino + + bool setDCMultiplierDivisor(ZIGBEE_DC_MEASUREMENT_TYPE measurement_type, uint16_t multiplier, uint16_t divisor); + +* ``measurement_type`` - DC measurement type constant (ZIGBEE_DC_MEASUREMENT_TYPE_VOLTAGE, ZIGBEE_DC_MEASUREMENT_TYPE_CURRENT, ZIGBEE_DC_MEASUREMENT_TYPE_POWER) +* ``multiplier`` - Multiplier value +* ``divisor`` - Divisor value + +This function will return ``true`` if successful, ``false`` otherwise. + +setDCReporting +^^^^^^^^^^^^^^ + +Sets the reporting configuration for DC measurements. + +.. code-block:: arduino + + bool setDCReporting(ZIGBEE_DC_MEASUREMENT_TYPE measurement_type, uint16_t min_interval, uint16_t max_interval, int16_t delta); + +* ``measurement_type`` - DC measurement type constant (ZIGBEE_DC_MEASUREMENT_TYPE_VOLTAGE, ZIGBEE_DC_MEASUREMENT_TYPE_CURRENT, ZIGBEE_DC_MEASUREMENT_TYPE_POWER) +* ``min_interval`` - Minimum reporting interval in seconds +* ``max_interval`` - Maximum reporting interval in seconds +* ``delta`` - Minimum change required to trigger a report + +This function will return ``true`` if successful, ``false`` otherwise. + +reportDC +^^^^^^^^ + +Manually reports a DC measurement value. + +.. code-block:: arduino + + bool reportDC(ZIGBEE_DC_MEASUREMENT_TYPE measurement_type); + +* ``measurement_type`` - DC measurement type constant (ZIGBEE_DC_MEASUREMENT_TYPE_VOLTAGE, ZIGBEE_DC_MEASUREMENT_TYPE_CURRENT, ZIGBEE_DC_MEASUREMENT_TYPE_POWER) + +This function will return ``true`` if successful, ``false`` otherwise. + +AC Electrical Measurement +************************* + +addACMeasurement +^^^^^^^^^^^^^^^^ + +Adds an AC measurement type for a specific phase. + +.. code-block:: arduino + + bool addACMeasurement(ZIGBEE_AC_MEASUREMENT_TYPE measurement_type, ZIGBEE_AC_PHASE_TYPE phase_type); + +* ``measurement_type`` - AC measurement type constant (ZIGBEE_AC_MEASUREMENT_TYPE_VOLTAGE, ZIGBEE_AC_MEASUREMENT_TYPE_CURRENT, ZIGBEE_AC_MEASUREMENT_TYPE_POWER, ZIGBEE_AC_MEASUREMENT_TYPE_POWER_FACTOR, ZIGBEE_AC_MEASUREMENT_TYPE_FREQUENCY) +* ``phase_type`` - Phase type constant (ZIGBEE_AC_PHASE_TYPE_NON_SPECIFIC, ZIGBEE_AC_PHASE_TYPE_A, ZIGBEE_AC_PHASE_TYPE_B, ZIGBEE_AC_PHASE_TYPE_C) + +This function will return ``true`` if successful, ``false`` otherwise. + +setACMeasurement +^^^^^^^^^^^^^^^^ + +Sets the AC measurement value for a specific measurement type and phase. + +.. code-block:: arduino + + bool setACMeasurement(ZIGBEE_AC_MEASUREMENT_TYPE measurement_type, ZIGBEE_AC_PHASE_TYPE phase_type, int32_t value); + +* ``measurement_type`` - AC measurement type constant (ZIGBEE_AC_MEASUREMENT_TYPE_VOLTAGE, ZIGBEE_AC_MEASUREMENT_TYPE_CURRENT, ZIGBEE_AC_MEASUREMENT_TYPE_POWER, ZIGBEE_AC_MEASUREMENT_TYPE_POWER_FACTOR, ZIGBEE_AC_MEASUREMENT_TYPE_FREQUENCY) +* ``phase_type`` - Phase type constant (ZIGBEE_AC_PHASE_TYPE_NON_SPECIFIC, ZIGBEE_AC_PHASE_TYPE_A, ZIGBEE_AC_PHASE_TYPE_B, ZIGBEE_AC_PHASE_TYPE_C) +* ``value`` - Measurement value + +This function will return ``true`` if successful, ``false`` otherwise. + +setACMinMaxValue +^^^^^^^^^^^^^^^^ + +Sets the minimum and maximum values for an AC measurement type and phase. + +.. code-block:: arduino + + bool setACMinMaxValue(ZIGBEE_AC_MEASUREMENT_TYPE measurement_type, ZIGBEE_AC_PHASE_TYPE phase_type, int32_t min, int32_t max); + +* ``measurement_type`` - AC measurement type constant (ZIGBEE_AC_MEASUREMENT_TYPE_VOLTAGE, ZIGBEE_AC_MEASUREMENT_TYPE_CURRENT, ZIGBEE_AC_MEASUREMENT_TYPE_POWER, ZIGBEE_AC_MEASUREMENT_TYPE_POWER_FACTOR, ZIGBEE_AC_MEASUREMENT_TYPE_FREQUENCY) +* ``phase_type`` - Phase type constant (ZIGBEE_AC_PHASE_TYPE_NON_SPECIFIC, ZIGBEE_AC_PHASE_TYPE_A, ZIGBEE_AC_PHASE_TYPE_B, ZIGBEE_AC_PHASE_TYPE_C) +* ``min`` - Minimum value +* ``max`` - Maximum value + +This function will return ``true`` if successful, ``false`` otherwise. + +setACMultiplierDivisor +^^^^^^^^^^^^^^^^^^^^^^ + +Sets the multiplier and divisor for scaling AC measurements. + +.. code-block:: arduino + + bool setACMultiplierDivisor(ZIGBEE_AC_MEASUREMENT_TYPE measurement_type, uint16_t multiplier, uint16_t divisor); + +* ``measurement_type`` - AC measurement type constant (ZIGBEE_AC_MEASUREMENT_TYPE_VOLTAGE, ZIGBEE_AC_MEASUREMENT_TYPE_CURRENT, ZIGBEE_AC_MEASUREMENT_TYPE_POWER, ZIGBEE_AC_MEASUREMENT_TYPE_POWER_FACTOR, ZIGBEE_AC_MEASUREMENT_TYPE_FREQUENCY) +* ``multiplier`` - Multiplier value +* ``divisor`` - Divisor value + +This function will return ``true`` if successful, ``false`` otherwise. + +setACPowerFactor +^^^^^^^^^^^^^^^^ + +Sets the power factor for a specific phase. + +.. code-block:: arduino + + bool setACPowerFactor(ZIGBEE_AC_PHASE_TYPE phase_type, int8_t power_factor); + +* ``phase_type`` - Phase type constant (ZIGBEE_AC_PHASE_TYPE_NON_SPECIFIC, ZIGBEE_AC_PHASE_TYPE_A, ZIGBEE_AC_PHASE_TYPE_B, ZIGBEE_AC_PHASE_TYPE_C) +* ``power_factor`` - Power factor value (-100 to 100) + +This function will return ``true`` if successful, ``false`` otherwise. + +setACReporting +^^^^^^^^^^^^^^ + +Sets the reporting configuration for AC measurements. + +.. code-block:: arduino + + bool setACReporting(ZIGBEE_AC_MEASUREMENT_TYPE measurement_type, ZIGBEE_AC_PHASE_TYPE phase_type, uint16_t min_interval, uint16_t max_interval, int32_t delta); + +* ``measurement_type`` - AC measurement type constant (ZIGBEE_AC_MEASUREMENT_TYPE_VOLTAGE, ZIGBEE_AC_MEASUREMENT_TYPE_CURRENT, ZIGBEE_AC_MEASUREMENT_TYPE_POWER, ZIGBEE_AC_MEASUREMENT_TYPE_POWER_FACTOR, ZIGBEE_AC_MEASUREMENT_TYPE_FREQUENCY) +* ``phase_type`` - Phase type constant (ZIGBEE_AC_PHASE_TYPE_NON_SPECIFIC, ZIGBEE_AC_PHASE_TYPE_A, ZIGBEE_AC_PHASE_TYPE_B, ZIGBEE_AC_PHASE_TYPE_C) +* ``min_interval`` - Minimum reporting interval in seconds +* ``max_interval`` - Maximum reporting interval in seconds +* ``delta`` - Minimum change required to trigger a report + +This function will return ``true`` if successful, ``false`` otherwise. + +reportAC +^^^^^^^^ + +Manually reports an AC measurement value. + +.. code-block:: arduino + + bool reportAC(ZIGBEE_AC_MEASUREMENT_TYPE measurement_type, ZIGBEE_AC_PHASE_TYPE phase_type); + +* ``measurement_type`` - AC measurement type constant (ZIGBEE_AC_MEASUREMENT_TYPE_VOLTAGE, ZIGBEE_AC_MEASUREMENT_TYPE_CURRENT, ZIGBEE_AC_MEASUREMENT_TYPE_POWER, ZIGBEE_AC_MEASUREMENT_TYPE_POWER_FACTOR, ZIGBEE_AC_MEASUREMENT_TYPE_FREQUENCY) +* ``phase_type`` - Phase type constant (ZIGBEE_AC_PHASE_TYPE_NON_SPECIFIC, ZIGBEE_AC_PHASE_TYPE_A, ZIGBEE_AC_PHASE_TYPE_B, ZIGBEE_AC_PHASE_TYPE_C) + +This function will return ``true`` if successful, ``false`` otherwise. + +Example +------- + +DC Electrical Measurement +************************* + +.. literalinclude:: ../../../libraries/Zigbee/examples/Zigbee_Electrical_DC_Sensor/Zigbee_Electrical_DC_Sensor.ino + :language: arduino + +AC Electrical Measurement +************************* + +.. literalinclude:: ../../../libraries/Zigbee/examples/Zigbee_Electrical_AC_Sensor_MultiPhase/Zigbee_Electrical_AC_Sensor_MultiPhase.ino + :language: arduino diff --git a/docs/en/zigbee/ep_flow_sensor.rst b/docs/en/zigbee/ep_flow_sensor.rst new file mode 100644 index 00000000000..9423f321a5d --- /dev/null +++ b/docs/en/zigbee/ep_flow_sensor.rst @@ -0,0 +1,119 @@ +################ +ZigbeeFlowSensor +################ + +About +----- + +The ``ZigbeeFlowSensor`` class provides a flow sensor endpoint for Zigbee networks. This endpoint implements the Zigbee Home Automation (HA) standard for liquid and gas flow measurement devices. + +**Features:** +* Flow rate measurement in m³/h +* Configurable measurement range +* Tolerance and reporting configuration +* Automatic reporting capabilities +* Integration with common endpoint features (binding, OTA, etc.) +* Zigbee HA standard compliance + +**Use Cases:** +* Water flow monitoring +* Gas flow measurement +* Industrial process monitoring +* Smart home water management +* HVAC system flow monitoring +* Agricultural irrigation systems + +API Reference +------------- + +Constructor +*********** + +ZigbeeFlowSensor +^^^^^^^^^^^^^^^^ + +Creates a new Zigbee flow sensor endpoint. + +.. code-block:: arduino + + ZigbeeFlowSensor(uint8_t endpoint); + +* ``endpoint`` - Endpoint number (1-254) + +API Methods +*********** + +setFlow +^^^^^^^ + +Sets the flow rate measurement value. + +.. code-block:: arduino + + bool setFlow(float value); + +* ``value`` - Flow rate value in 0.1 m³/h + +This function will return ``true`` if successful, ``false`` otherwise. + +setMinMaxValue +^^^^^^^^^^^^^^ + +Sets the minimum and maximum measurement values. + +.. code-block:: arduino + + bool setMinMaxValue(float min, float max); + +* ``min`` - Minimum flow rate value in 0.1 m³/h +* ``max`` - Maximum flow rate value in 0.1 m³/h + +This function will return ``true`` if successful, ``false`` otherwise. + +setTolerance +^^^^^^^^^^^^ + +Sets the tolerance value for measurements. + +.. code-block:: arduino + + bool setTolerance(float tolerance); + +* ``tolerance`` - Tolerance value in 0.01 m³/h + +This function will return ``true`` if successful, ``false`` otherwise. + +setReporting +^^^^^^^^^^^^ + +Sets the reporting configuration for flow rate measurements. + +.. code-block:: arduino + + bool setReporting(uint16_t min_interval, uint16_t max_interval, float delta); + +* ``min_interval`` - Minimum reporting interval in seconds +* ``max_interval`` - Maximum reporting interval in seconds +* ``delta`` - Minimum change required to trigger a report in 0.1 m³/h + +This function will return ``true`` if successful, ``false`` otherwise. + +report +^^^^^^ + +Manually reports the current flow rate value. + +.. code-block:: arduino + + bool report(); + +This function will return ``true`` if successful, ``false`` otherwise. + +Example +------- + +Flow + PressureSensor Implementation +************************************ + +.. literalinclude:: ../../../libraries/Zigbee/examples/Zigbee_Pressure_Flow_Sensor/Zigbee_Pressure_Flow_Sensor.ino + :language: arduino diff --git a/docs/en/zigbee/ep_gateway.rst b/docs/en/zigbee/ep_gateway.rst new file mode 100644 index 00000000000..d436887a373 --- /dev/null +++ b/docs/en/zigbee/ep_gateway.rst @@ -0,0 +1,35 @@ +############# +ZigbeeGateway +############# + +About +----- + +The ``ZigbeeGateway`` class provides a gateway endpoint for Zigbee networks. This endpoint implements the Zigbee Home Automation (HA) standard for network coordination and gateway functionality. +Gateway is a device that can be used to bridge Zigbee network to other networks (e.g. Wi-Fi, Ethernet, etc.). + +API Reference +------------- + +Constructor +*********** + +ZigbeeGateway +^^^^^^^^^^^^^ + +Creates a new Zigbee gateway endpoint. + +.. code-block:: arduino + + ZigbeeGateway(uint8_t endpoint); + +* ``endpoint`` - Endpoint number (1-254) + +Example +------- + +Gateway Implementation +********************** + +.. literalinclude:: ../../../libraries/Zigbee/examples/Zigbee_Gateway/Zigbee_Gateway.ino + :language: arduino diff --git a/docs/en/zigbee/ep_illuminance_sensor.rst b/docs/en/zigbee/ep_illuminance_sensor.rst new file mode 100644 index 00000000000..1e627f7dfe9 --- /dev/null +++ b/docs/en/zigbee/ep_illuminance_sensor.rst @@ -0,0 +1,109 @@ +####################### +ZigbeeIlluminanceSensor +####################### + +About +----- + +The ``ZigbeeIlluminanceSensor`` class provides an endpoint for illuminance sensors in Zigbee networks. This endpoint implements the Zigbee Home Automation (HA) standard for light level measurement devices, supporting ambient light monitoring. + +**Features:** +* Illuminance measurement in lux +* Configurable measurement range +* Tolerance and reporting configuration +* Automatic reporting capabilities + +API Reference +------------- + +Constructor +*********** + +ZigbeeIlluminanceSensor +^^^^^^^^^^^^^^^^^^^^^^^ + +Creates a new Zigbee illuminance sensor endpoint. + +.. code-block:: arduino + + ZigbeeIlluminanceSensor(uint8_t endpoint); + +* ``endpoint`` - Endpoint number (1-254) + +API Methods +*********** + +setIlluminance +^^^^^^^^^^^^^^ + +Sets the illuminance measurement value. + +.. code-block:: arduino + + bool setIlluminance(uint16_t value); + +* ``value`` - Illuminance value in lux + +This function will return ``true`` if successful, ``false`` otherwise. + +setMinMaxValue +^^^^^^^^^^^^^^ + +Sets the minimum and maximum measurement values. + +.. code-block:: arduino + + bool setMinMaxValue(uint16_t min, uint16_t max); + +* ``min`` - Minimum illuminance value in lux +* ``max`` - Maximum illuminance value in lux + +This function will return ``true`` if successful, ``false`` otherwise. + +setTolerance +^^^^^^^^^^^^ + +Sets the tolerance value for measurements. + +.. code-block:: arduino + + bool setTolerance(uint16_t tolerance); + +* ``tolerance`` - Tolerance value in lux + +This function will return ``true`` if successful, ``false`` otherwise. + +setReporting +^^^^^^^^^^^^ + +Sets the reporting configuration for illuminance measurements. + +.. code-block:: arduino + + bool setReporting(uint16_t min_interval, uint16_t max_interval, uint16_t delta); + +* ``min_interval`` - Minimum reporting interval in seconds +* ``max_interval`` - Maximum reporting interval in seconds +* ``delta`` - Minimum change required to trigger a report in lux + +This function will return ``true`` if successful, ``false`` otherwise. + +report +^^^^^^ + +Manually reports the current illuminance value. + +.. code-block:: arduino + + bool report(); + +This function will return ``true`` if successful, ``false`` otherwise. + +Example +------- + +Illuminance Sensor Implementation +********************************* + +.. literalinclude:: ../../../libraries/Zigbee/examples/Zigbee_Illuminance_Sensor/Zigbee_Illuminance_Sensor.ino + :language: arduino diff --git a/docs/en/zigbee/ep_light.rst b/docs/en/zigbee/ep_light.rst new file mode 100644 index 00000000000..fb70320321b --- /dev/null +++ b/docs/en/zigbee/ep_light.rst @@ -0,0 +1,88 @@ +########### +ZigbeeLight +########### + +About +----- + +The ``ZigbeeLight`` class provides a simple on/off light endpoint for Zigbee networks. This endpoint implements the Zigbee Home Automation (HA) standard for basic lighting control. + +**Features:** +* Simple on/off control +* State change callbacks + +API Reference +------------- + +Constructor +*********** + +ZigbeeLight +^^^^^^^^^^^ + +Creates a new Zigbee light endpoint. + +.. code-block:: arduino + + ZigbeeLight(uint8_t endpoint); + +* ``endpoint`` - Endpoint number (1-254) + +Light Control +************* + +setLight +^^^^^^^^ + +Sets the light state (on or off). + +.. code-block:: arduino + + bool setLight(bool state); + +* ``state`` - ``true`` to turn on, ``false`` to turn off + +This function will return ``true`` if successful, ``false`` otherwise. + +getLightState +^^^^^^^^^^^^^ + +Gets the current light state. + +.. code-block:: arduino + + bool getLightState(); + +This function will return current light state (``true`` = on, ``false`` = off). + +restoreLight +^^^^^^^^^^^^ + +Restores the light state and triggers any registered callbacks. + +.. code-block:: arduino + + void restoreLight(); + +Event Handling +************** + +onLightChange +^^^^^^^^^^^^^ + +Sets a callback function to be called when the light state changes. + +.. code-block:: arduino + + void onLightChange(void (*callback)(bool)); + +* ``callback`` - Function to call when light state changes + +Example +------- + +Basic Light Implementation +************************** + +.. literalinclude:: ../../../libraries/Zigbee/examples/Zigbee_On_Off_Light/Zigbee_On_Off_Light.ino + :language: arduino diff --git a/docs/en/zigbee/ep_occupancy_sensor.rst b/docs/en/zigbee/ep_occupancy_sensor.rst new file mode 100644 index 00000000000..7fe50d59ea4 --- /dev/null +++ b/docs/en/zigbee/ep_occupancy_sensor.rst @@ -0,0 +1,81 @@ +##################### +ZigbeeOccupancySensor +##################### + +About +----- + +The ``ZigbeeOccupancySensor`` class provides an endpoint for occupancy sensors in Zigbee networks. This endpoint implements the Zigbee Home Automation (HA) standard for occupancy detection devices, supporting various sensor types for detecting presence. + +**Features:** +* Occupancy detection (occupied/unoccupied) +* Multiple sensor type support (PIR, ultrasonic, etc.) +* Automatic reporting capabilities +* Integration with common endpoint features (binding, OTA, etc.) +* Zigbee HA standard compliance + +API Reference +------------- + +Constructor +*********** + +ZigbeeOccupancySensor +^^^^^^^^^^^^^^^^^^^^^ + +Creates a new Zigbee occupancy sensor endpoint. + +.. code-block:: arduino + + ZigbeeOccupancySensor(uint8_t endpoint); + +* ``endpoint`` - Endpoint number (1-254) + +API Methods +*********** + +setOccupancy +^^^^^^^^^^^^ + +Sets the occupancy state. + +.. code-block:: arduino + + bool setOccupancy(bool occupied); + +* ``occupied`` - Occupancy state (true = occupied, false = unoccupied) + +This function will return ``true`` if successful, ``false`` otherwise. + +setSensorType +^^^^^^^^^^^^^ + +Sets the sensor type. + +.. code-block:: arduino + + bool setSensorType(uint8_t sensor_type); + +* ``sensor_type`` - Sensor type identifier (see esp_zb_zcl_occupancy_sensing_occupancy_sensor_type_t) + +This function will return ``true`` if successful, ``false`` otherwise. + +report +^^^^^^ + +Manually reports the current occupancy state. + +.. code-block:: arduino + + bool report(); + +This function will return ``true`` if successful, ``false`` otherwise. + +Example +------- + +Occupancy Sensor Implementation +******************************* + +.. literalinclude:: ../../../libraries/Zigbee/examples/Zigbee_Occupancy_Sensor/Zigbee_Occupancy_Sensor.ino + :language: arduino diff --git a/docs/en/zigbee/ep_pm25_sensor.rst b/docs/en/zigbee/ep_pm25_sensor.rst new file mode 100644 index 00000000000..2f1432f8224 --- /dev/null +++ b/docs/en/zigbee/ep_pm25_sensor.rst @@ -0,0 +1,109 @@ +################ +ZigbeePM25Sensor +################ + +About +----- + +The ``ZigbeePM25Sensor`` class provides a PM2.5 air quality sensor endpoint for Zigbee networks. This endpoint implements the Zigbee Home Automation (HA) standard for particulate matter measurement devices. + +**Features:** +* PM2.5 concentration measurement in μg/m³ +* Configurable measurement range +* Tolerance and reporting configuration +* Automatic reporting capabilities + +API Reference +------------- + +Constructor +*********** + +ZigbeePM25Sensor +^^^^^^^^^^^^^^^^ + +Creates a new Zigbee PM2.5 sensor endpoint. + +.. code-block:: arduino + + ZigbeePM25Sensor(uint8_t endpoint); + +* ``endpoint`` - Endpoint number (1-254) + +API Methods +*********** + +setPM25 +^^^^^^^ + +Sets the PM2.5 concentration measurement value. + +.. code-block:: arduino + + bool setPM25(float pm25); + +* ``pm25`` - PM2.5 concentration value in 0.1 μg/m³ + +This function will return ``true`` if successful, ``false`` otherwise. + +setMinMaxValue +^^^^^^^^^^^^^^ + +Sets the minimum and maximum measurement values. + +.. code-block:: arduino + + bool setMinMaxValue(float min, float max); + +* ``min`` - Minimum PM2.5 concentration value in 0.1 μg/m³ +* ``max`` - Maximum PM2.5 concentration value in 0.1 μg/m³ + +This function will return ``true`` if successful, ``false`` otherwise. + +setTolerance +^^^^^^^^^^^^ + +Sets the tolerance value for measurements. + +.. code-block:: arduino + + bool setTolerance(float tolerance); + +* ``tolerance`` - Tolerance value in 0.1 μg/m³ + +This function will return ``true`` if successful, ``false`` otherwise. + +setReporting +^^^^^^^^^^^^ + +Sets the reporting configuration for PM2.5 measurements. + +.. code-block:: arduino + + bool setReporting(uint16_t min_interval, uint16_t max_interval, float delta); + +* ``min_interval`` - Minimum reporting interval in seconds +* ``max_interval`` - Maximum reporting interval in seconds +* ``delta`` - Minimum change required to trigger a report in 0.1 μg/m³ + +This function will return ``true`` if successful, ``false`` otherwise. + +report +^^^^^^ + +Manually reports the current PM2.5 concentration value. + +.. code-block:: arduino + + bool report(); + +This function will return ``true`` if successful, ``false`` otherwise. + +Example +------- + +PM2.5 Sensor Implementation +*************************** + +.. literalinclude:: ../../../libraries/Zigbee/examples/Zigbee_PM25_Sensor/Zigbee_PM25_Sensor.ino + :language: arduino diff --git a/docs/en/zigbee/ep_power_outlet.rst b/docs/en/zigbee/ep_power_outlet.rst new file mode 100644 index 00000000000..97b22cbbc2a --- /dev/null +++ b/docs/en/zigbee/ep_power_outlet.rst @@ -0,0 +1,88 @@ +################# +ZigbeePowerOutlet +################# + +About +----- + +The ``ZigbeePowerOutlet`` class provides a smart power outlet endpoint for Zigbee networks. This endpoint implements the Zigbee Home Automation (HA) standard for power control, allowing remote on/off control of electrical devices. + +**Features:** +* On/off power control +* State change callbacks + +API Reference +------------- + +Constructor +*********** + +ZigbeePowerOutlet +^^^^^^^^^^^^^^^^^ + +Creates a new Zigbee power outlet endpoint. + +.. code-block:: arduino + + ZigbeePowerOutlet(uint8_t endpoint); + +* ``endpoint`` - Endpoint number (1-254) + +Power Control +************* + +setState +^^^^^^^^ + +Sets the power outlet state (on or off). + +.. code-block:: arduino + + bool setState(bool state); + +* ``state`` - ``true`` to turn on, ``false`` to turn off + +This function will return ``true`` if successful, ``false`` otherwise. + +getPowerOutletState +^^^^^^^^^^^^^^^^^^^ + +Gets the current power outlet state. + +.. code-block:: arduino + + bool getPowerOutletState(); + +This function will return current power state (``true`` = on, ``false`` = off). + +restoreState +^^^^^^^^^^^^ + +Restores the power outlet state and triggers any registered callbacks. + +.. code-block:: arduino + + void restoreState(); + +Event Handling +************** + +onPowerOutletChange +^^^^^^^^^^^^^^^^^^^ + +Sets a callback function to be called when the power outlet state changes. + +.. code-block:: arduino + + void onPowerOutletChange(void (*callback)(bool)); + +* ``callback`` - Function to call when power outlet state changes + +Example +------- + +Smart Power Outlet Implementation +********************************* + +.. literalinclude:: ../../../libraries/Zigbee/examples/Zigbee_Power_Outlet/Zigbee_Power_Outlet.ino + :language: arduino diff --git a/docs/en/zigbee/ep_pressure_sensor.rst b/docs/en/zigbee/ep_pressure_sensor.rst new file mode 100644 index 00000000000..5d857bb163e --- /dev/null +++ b/docs/en/zigbee/ep_pressure_sensor.rst @@ -0,0 +1,109 @@ +#################### +ZigbeePressureSensor +#################### + +About +----- + +The ``ZigbeePressureSensor`` class provides an endpoint for pressure sensors in Zigbee networks. This endpoint implements the Zigbee Home Automation (HA) standard for pressure measurement devices, supporting atmospheric pressure, barometric pressure, and other pressure measurements. + +**Features:** +* Pressure measurement in hPa (hectopascals) +* Configurable measurement range +* Tolerance and reporting configuration +* Automatic reporting capabilities + +API Reference +------------- + +Constructor +*********** + +ZigbeePressureSensor +^^^^^^^^^^^^^^^^^^^^ + +Creates a new Zigbee pressure sensor endpoint. + +.. code-block:: arduino + + ZigbeePressureSensor(uint8_t endpoint); + +* ``endpoint`` - Endpoint number (1-254) + +API Methods +*********** + +setPressure +^^^^^^^^^^^ + +Sets the pressure measurement value. + +.. code-block:: arduino + + bool setPressure(int16_t value); + +* ``value`` - Pressure value in hPa + +This function will return ``true`` if successful, ``false`` otherwise. + +setMinMaxValue +^^^^^^^^^^^^^^ + +Sets the minimum and maximum measurement values. + +.. code-block:: arduino + + bool setMinMaxValue(int16_t min, int16_t max); + +* ``min`` - Minimum pressure value in hPa +* ``max`` - Maximum pressure value in hPa + +This function will return ``true`` if successful, ``false`` otherwise. + +setTolerance +^^^^^^^^^^^^ + +Sets the tolerance value for measurements. + +.. code-block:: arduino + + bool setTolerance(uint16_t tolerance); + +* ``tolerance`` - Tolerance value in hPa + +This function will return ``true`` if successful, ``false`` otherwise. + +setReporting +^^^^^^^^^^^^ + +Sets the reporting configuration for pressure measurements. + +.. code-block:: arduino + + bool setReporting(uint16_t min_interval, uint16_t max_interval, uint16_t delta); + +* ``min_interval`` - Minimum reporting interval in seconds +* ``max_interval`` - Maximum reporting interval in seconds +* ``delta`` - Minimum change required to trigger a report in hPa + +This function will return ``true`` if successful, ``false`` otherwise. + +report +^^^^^^ + +Manually reports the current pressure value. + +.. code-block:: arduino + + bool report(); + +This function will return ``true`` if successful, ``false`` otherwise. + +Example +------- + +Pressure + Flow Sensor Implementation +************************************* + +.. literalinclude:: ../../../libraries/Zigbee/examples/Zigbee_Pressure_Flow_Sensor/Zigbee_Pressure_Flow_Sensor.ino + :language: arduino diff --git a/docs/en/zigbee/ep_range_extender.rst b/docs/en/zigbee/ep_range_extender.rst new file mode 100644 index 00000000000..b451f8764fc --- /dev/null +++ b/docs/en/zigbee/ep_range_extender.rst @@ -0,0 +1,34 @@ +################### +ZigbeeRangeExtender +################### + +About +----- + +The ``ZigbeeRangeExtender`` class provides a range extender endpoint for Zigbee networks. This endpoint implements the Zigbee Home Automation (HA) standard for range extender devices that help extend the coverage of Zigbee networks by acting as repeaters. + +API Reference +------------- + +Constructor +*********** + +ZigbeeRangeExtender +^^^^^^^^^^^^^^^^^^^ + +Creates a new Zigbee range extender endpoint. + +.. code-block:: arduino + + ZigbeeRangeExtender(uint8_t endpoint); + +* ``endpoint`` - Endpoint number (1-254) + +Example +------- + +Range Extender Implementation +***************************** + +.. literalinclude:: ../../../libraries/Zigbee/examples/Zigbee_Range_Extender/Zigbee_Range_Extender.ino + :language: arduino diff --git a/docs/en/zigbee/ep_switch.rst b/docs/en/zigbee/ep_switch.rst new file mode 100644 index 00000000000..e1847996cc2 --- /dev/null +++ b/docs/en/zigbee/ep_switch.rst @@ -0,0 +1,136 @@ +############ +ZigbeeSwitch +############ + +About +----- + +The ``ZigbeeSwitch`` class provides a switch endpoint for Zigbee networks. This endpoint implements the Zigbee Home Automation (HA) standard for controlling other devices (typically lights) through on/off commands. + +**Features:** +* On/off control commands for bound devices +* Group control support +* Direct device addressing + +API Reference +------------- + +Constructor +*********** + +ZigbeeSwitch +^^^^^^^^^^^^ + +Creates a new Zigbee switch endpoint. + +.. code-block:: arduino + + ZigbeeSwitch(uint8_t endpoint); + +* ``endpoint`` - Endpoint number (1-254) + +Basic Control Commands +********************** + +lightToggle +^^^^^^^^^^^ + +Toggles the state of bound lights (on to off, or off to on). + +.. code-block:: arduino + + void lightToggle(); + void lightToggle(uint16_t group_addr); + void lightToggle(uint8_t endpoint, uint16_t short_addr); + void lightToggle(uint8_t endpoint, esp_zb_ieee_addr_t ieee_addr); + +* ``group_addr`` - Group address to control (optional) +* ``endpoint`` - Target device endpoint (optional) +* ``short_addr`` - Target device short address (optional) +* ``ieee_addr`` - Target device IEEE address (optional) + +lightOn +^^^^^^^ + +Turns on bound lights. + +.. code-block:: arduino + + void lightOn(); + void lightOn(uint16_t group_addr); + void lightOn(uint8_t endpoint, uint16_t short_addr); + void lightOn(uint8_t endpoint, esp_zb_ieee_addr_t ieee_addr); + +* ``group_addr`` - Group address to control (optional) +* ``endpoint`` - Target device endpoint (optional) +* ``short_addr`` - Target device short address (optional) +* ``ieee_addr`` - Target device IEEE address (optional) + +lightOff +^^^^^^^^ + +Turns off bound lights. + +.. code-block:: arduino + + void lightOff(); + void lightOff(uint16_t group_addr); + void lightOff(uint8_t endpoint, uint16_t short_addr); + void lightOff(uint8_t endpoint, esp_zb_ieee_addr_t ieee_addr); + +* ``group_addr`` - Group address to control (optional) +* ``endpoint`` - Target device endpoint (optional) +* ``short_addr`` - Target device short address (optional) +* ``ieee_addr`` - Target device IEEE address (optional) + +Advanced Control Commands +************************* + +lightOffWithEffect +^^^^^^^^^^^^^^^^^^ + +Turns off lights with a specific effect. + +.. code-block:: arduino + + void lightOffWithEffect(uint8_t effect_id, uint8_t effect_variant); + +* ``effect_id`` - Effect identifier +* ``effect_variant`` - Effect variant + +lightOnWithTimedOff +^^^^^^^^^^^^^^^^^^^ + +Turns on lights with automatic turn-off after specified time. + +.. code-block:: arduino + + void lightOnWithTimedOff(uint8_t on_off_control, uint16_t time_on, uint16_t time_off); + +* ``on_off_control`` - Control byte +* ``time_on`` - Time to stay on (in 1/10th seconds) +* ``time_off`` - Time to stay off (in 1/10th seconds) + +lightOnWithSceneRecall +^^^^^^^^^^^^^^^^^^^^^^ + +Turns on lights by recalling a scene. + +.. code-block:: arduino + + void lightOnWithSceneRecall(); + +Example +------- + +Basic Switch Implementation +*************************** + +.. literalinclude:: ../../../libraries/Zigbee/examples/Zigbee_On_Off_Switch/Zigbee_On_Off_Switch.ino + :language: arduino + +Multi Switch Implementation +*************************** + +.. literalinclude:: ../../../libraries/Zigbee/examples/Zigbee_On_Off_MultiSwitch/Zigbee_On_Off_MultiSwitch.ino + :language: arduino diff --git a/docs/en/zigbee/ep_temperature_sensor.rst b/docs/en/zigbee/ep_temperature_sensor.rst new file mode 100644 index 00000000000..85c2147f1a1 --- /dev/null +++ b/docs/en/zigbee/ep_temperature_sensor.rst @@ -0,0 +1,184 @@ +################ +ZigbeeTempSensor +################ + +About +----- + +The ``ZigbeeTempSensor`` class provides a temperature and humidity sensor endpoint for Zigbee networks. This endpoint implements the Zigbee Home Automation (HA) standard for environmental monitoring with configurable reporting intervals and thresholds. + +**Features:** +* Temperature measurement and reporting +* Optional humidity measurement +* Configurable reporting intervals +* Min/max value and tolerance settings + +API Reference +------------- + +Constructor +*********** + +ZigbeeTempSensor +^^^^^^^^^^^^^^^^ + +Creates a new Zigbee temperature sensor endpoint. + +.. code-block:: arduino + + ZigbeeTempSensor(uint8_t endpoint); + +* ``endpoint`` - Endpoint number (1-254) + +Temperature Control +******************* + +setTemperature +^^^^^^^^^^^^^^ + +Sets the temperature value in 0.01°C resolution. + +.. code-block:: arduino + + bool setTemperature(float value); + +* ``value`` - Temperature value in degrees Celsius + +This function will return ``true`` if successful, ``false`` otherwise. + +setMinMaxValue +^^^^^^^^^^^^^^ + +Sets the minimum and maximum temperature values for the sensor. + +.. code-block:: arduino + + bool setMinMaxValue(float min, float max); + +* ``min`` - Minimum temperature value in degrees Celsius +* ``max`` - Maximum temperature value in degrees Celsius + +This function will return ``true`` if successful, ``false`` otherwise. + +setTolerance +^^^^^^^^^^^^ + +Sets the tolerance value for temperature reporting. + +.. code-block:: arduino + + bool setTolerance(float tolerance); + +* ``tolerance`` - Tolerance value in degrees Celsius + +This function will return ``true`` if successful, ``false`` otherwise. + +setReporting +^^^^^^^^^^^^ + +Sets the reporting interval for temperature measurements. + +.. code-block:: arduino + + bool setReporting(uint16_t min_interval, uint16_t max_interval, float delta); + +* ``min_interval`` - Minimum reporting interval in seconds +* ``max_interval`` - Maximum reporting interval in seconds +* ``delta`` - Minimum change in temperature to trigger report (in 0.01°C) + +This function will return ``true`` if successful, ``false`` otherwise. + +reportTemperature +^^^^^^^^^^^^^^^^^ + +Manually reports the current temperature value. + +.. code-block:: arduino + + bool reportTemperature(); + +This function will return ``true`` if successful, ``false`` otherwise. + +Humidity Control (Optional) +*************************** + +addHumiditySensor +^^^^^^^^^^^^^^^^^ + +Adds humidity measurement capability to the temperature sensor. + +.. code-block:: arduino + + void addHumiditySensor(float min, float max, float tolerance); + +* ``min`` - Minimum humidity value in percentage +* ``max`` - Maximum humidity value in percentage +* ``tolerance`` - Tolerance value in percentage + +setHumidity +^^^^^^^^^^^ + +Sets the humidity value in 0.01% resolution. + +.. code-block:: arduino + + bool setHumidity(float value); + +* ``value`` - Humidity value in percentage (0-100) + +This function will return ``true`` if successful, ``false`` otherwise. + +setHumidityReporting +^^^^^^^^^^^^^^^^^^^^ + +Sets the reporting interval for humidity measurements. + +.. code-block:: arduino + + bool setHumidityReporting(uint16_t min_interval, uint16_t max_interval, float delta); + +* ``min_interval`` - Minimum reporting interval in seconds +* ``max_interval`` - Maximum reporting interval in seconds +* ``delta`` - Minimum change in humidity to trigger report (in 0.01%) + +This function will return ``true`` if successful, ``false`` otherwise. + +reportHumidity +^^^^^^^^^^^^^^ + +Manually reports the current humidity value. + +.. code-block:: arduino + + bool reportHumidity(); + +This function will return ``true`` if successful, ``false`` otherwise. + +Combined Reporting +****************** + +report +^^^^^^ + +Reports both temperature and humidity values if humidity sensor is enabled. + +.. code-block:: arduino + + bool report(); + +This function will return ``true`` if successful, ``false`` otherwise. + +Example +------- + +Temperature Sensor Implementation +********************************* + +.. literalinclude:: ../../../libraries/Zigbee/examples/Zigbee_Temperature_Sensor/Zigbee_Temperature_Sensor.ino + :language: arduino + +Temperature + Humidity Sleepy Sensor Implementation +*************************************************** + +.. literalinclude:: ../../../libraries/Zigbee/examples/Zigbee_Temp_Hum_Sensor_Sleepy/Zigbee_Temp_Hum_Sensor_Sleepy.ino + :language: arduino diff --git a/docs/en/zigbee/ep_thermostat.rst b/docs/en/zigbee/ep_thermostat.rst new file mode 100644 index 00000000000..b7c79254d0f --- /dev/null +++ b/docs/en/zigbee/ep_thermostat.rst @@ -0,0 +1,238 @@ +################ +ZigbeeThermostat +################ + +About +----- + +The ``ZigbeeThermostat`` class provides a thermostat endpoint for Zigbee networks that receives temperature data from temperature sensors. This endpoint implements the Zigbee Home Automation (HA) standard for thermostats that can bind to temperature sensors and receive temperature readings. + +**Features:** +* Automatic discovery and binding to temperature sensors +* Temperature data reception from bound sensors +* Configurable temperature reporting intervals +* Sensor settings retrieval (min/max temperature, tolerance) +* Multiple addressing modes (group, specific endpoint, IEEE address) + +API Reference +------------- + +Constructor +*********** + +ZigbeeThermostat +^^^^^^^^^^^^^^^^ + +Creates a new Zigbee thermostat endpoint. + +.. code-block:: arduino + + ZigbeeThermostat(uint8_t endpoint); + +* ``endpoint`` - Endpoint number (1-254) + +Event Handling +************** + +onTempReceive +^^^^^^^^^^^^^ + +Sets a callback function for receiving temperature data. + +.. code-block:: arduino + + void onTempReceive(void (*callback)(float temperature)); + +* ``callback`` - Function to call when temperature data is received +* ``temperature`` - Temperature value in degrees Celsius + +onTempReceiveWithSource +^^^^^^^^^^^^^^^^^^^^^^^ + +Sets a callback function for receiving temperature data with source information. + +.. code-block:: arduino + + void onTempReceiveWithSource(void (*callback)(float temperature, uint8_t src_endpoint, esp_zb_zcl_addr_t src_address)); + +* ``callback`` - Function to call when temperature data is received +* ``temperature`` - Temperature value in degrees Celsius +* ``src_endpoint`` - Source endpoint that sent the temperature data +* ``src_address`` - Source address information + +onConfigReceive +^^^^^^^^^^^^^^^ + +Sets a callback function for receiving sensor configuration data. + +.. code-block:: arduino + + void onConfigReceive(void (*callback)(float min_temp, float max_temp, float tolerance)); + +* ``callback`` - Function to call when sensor configuration is received +* ``min_temp`` - Minimum temperature supported by the sensor +* ``max_temp`` - Maximum temperature supported by the sensor +* ``tolerance`` - Temperature tolerance of the sensor + +Temperature Data Retrieval +************************** + +getTemperature +^^^^^^^^^^^^^^ + +Requests temperature data from all bound sensors. + +.. code-block:: arduino + + void getTemperature(); + +getTemperature (Group) +^^^^^^^^^^^^^^^^^^^^^^ + +Requests temperature data from a specific group. + +.. code-block:: arduino + + void getTemperature(uint16_t group_addr); + +* ``group_addr`` - Group address to send the request to + +getTemperature (Endpoint + Short Address) +^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ + +Requests temperature data from a specific endpoint using short address. + +.. code-block:: arduino + + void getTemperature(uint8_t endpoint, uint16_t short_addr); + +* ``endpoint`` - Target endpoint number +* ``short_addr`` - Short address of the target device + +getTemperature (Endpoint + IEEE Address) +^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ + +Requests temperature data from a specific endpoint using IEEE address. + +.. code-block:: arduino + + void getTemperature(uint8_t endpoint, esp_zb_ieee_addr_t ieee_addr); + +* ``endpoint`` - Target endpoint number +* ``ieee_addr`` - IEEE address of the target device + +Sensor Settings Retrieval +************************* + +getSensorSettings +^^^^^^^^^^^^^^^^^ + +Requests sensor settings from all bound sensors. + +.. code-block:: arduino + + void getSensorSettings(); + +getSensorSettings (Group) +^^^^^^^^^^^^^^^^^^^^^^^^^ + +Requests sensor settings from a specific group. + +.. code-block:: arduino + + void getSensorSettings(uint16_t group_addr); + +* ``group_addr`` - Group address to send the request to + +getSensorSettings (Endpoint + Short Address) +^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ + +Requests sensor settings from a specific endpoint using short address. + +.. code-block:: arduino + + void getSensorSettings(uint8_t endpoint, uint16_t short_addr); + +* ``endpoint`` - Target endpoint number +* ``short_addr`` - Short address of the target device + +getSensorSettings (Endpoint + IEEE Address) +^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ + +Requests sensor settings from a specific endpoint using IEEE address. + +.. code-block:: arduino + + void getSensorSettings(uint8_t endpoint, esp_zb_ieee_addr_t ieee_addr); + +* ``endpoint`` - Target endpoint number +* ``ieee_addr`` - IEEE address of the target device + +Temperature Reporting Configuration +*********************************** + +setTemperatureReporting +^^^^^^^^^^^^^^^^^^^^^^^ + +Configures temperature reporting for all bound sensors. + +.. code-block:: arduino + + void setTemperatureReporting(uint16_t min_interval, uint16_t max_interval, float delta); + +* ``min_interval`` - Minimum reporting interval in seconds +* ``max_interval`` - Maximum reporting interval in seconds +* ``delta`` - Minimum change in temperature to trigger a report + +setTemperatureReporting (Group) +^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ + +Configures temperature reporting for a specific group. + +.. code-block:: arduino + + void setTemperatureReporting(uint16_t group_addr, uint16_t min_interval, uint16_t max_interval, float delta); + +* ``group_addr`` - Group address to configure +* ``min_interval`` - Minimum reporting interval in seconds +* ``max_interval`` - Maximum reporting interval in seconds +* ``delta`` - Minimum change in temperature to trigger a report + +setTemperatureReporting (Endpoint + Short Address) +^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ + +Configures temperature reporting for a specific endpoint using short address. + +.. code-block:: arduino + + void setTemperatureReporting(uint8_t endpoint, uint16_t short_addr, uint16_t min_interval, uint16_t max_interval, float delta); + +* ``endpoint`` - Target endpoint number +* ``short_addr`` - Short address of the target device +* ``min_interval`` - Minimum reporting interval in seconds +* ``max_interval`` - Maximum reporting interval in seconds +* ``delta`` - Minimum change in temperature to trigger a report + +setTemperatureReporting (Endpoint + IEEE Address) +^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ + +Configures temperature reporting for a specific endpoint using IEEE address. + +.. code-block:: arduino + + void setTemperatureReporting(uint8_t endpoint, esp_zb_ieee_addr_t ieee_addr, uint16_t min_interval, uint16_t max_interval, float delta); + +* ``endpoint`` - Target endpoint number +* ``ieee_addr`` - IEEE address of the target device +* ``min_interval`` - Minimum reporting interval in seconds +* ``max_interval`` - Maximum reporting interval in seconds +* ``delta`` - Minimum change in temperature to trigger a report + +Example +------- + +Thermostat Implementation +************************* + +.. literalinclude:: ../../../libraries/Zigbee/examples/Zigbee_Thermostat/Zigbee_Thermostat.ino + :language: arduino diff --git a/docs/en/zigbee/ep_vibration_sensor.rst b/docs/en/zigbee/ep_vibration_sensor.rst new file mode 100644 index 00000000000..896c4672c6d --- /dev/null +++ b/docs/en/zigbee/ep_vibration_sensor.rst @@ -0,0 +1,87 @@ +##################### +ZigbeeVibrationSensor +##################### + +About +----- + +The ``ZigbeeVibrationSensor`` class provides a vibration sensor endpoint for Zigbee networks. This endpoint implements the Zigbee Home Automation (HA) standard for vibration detection devices. + +**Features:** +* Vibration detection and measurement +* Configurable sensitivity levels +* Multiple detection modes +* Automatic reporting capabilities +* Integration with common endpoint features (binding, OTA, etc.) +* Zigbee HA standard compliance + +**Use Cases:** +* Security system vibration detection +* Industrial equipment monitoring +* Structural health monitoring +* Smart home security applications +* Machine condition monitoring + +API Reference +------------- + +Constructor +*********** + +ZigbeeVibrationSensor +^^^^^^^^^^^^^^^^^^^^^ + +Creates a new Zigbee vibration sensor endpoint. + +.. code-block:: arduino + + ZigbeeVibrationSensor(uint8_t endpoint); + +* ``endpoint`` - Endpoint number (1-254) + +API Methods +*********** + +setVibration +^^^^^^^^^^^^ + +Sets the vibration detection state. + +.. code-block:: arduino + + bool setVibration(bool sensed); + +* ``sensed`` - Vibration state (true = sensed, false = not sensed) + +This function will return ``true`` if successful, ``false`` otherwise. + +setIASClientEndpoint +^^^^^^^^^^^^^^^^^^^^ + +Sets the IAS Client endpoint number (default is 1). + +.. code-block:: arduino + + void setIASClientEndpoint(uint8_t ep_number); + +* ``ep_number`` - IAS Client endpoint number + +report +^^^^^^ + +Manually reports the current vibration state. + +.. code-block:: arduino + + void report(); + +This function does not return a value. + +Example +------- + +Vibration Sensor Implementation +******************************* + +.. literalinclude:: ../../../libraries/Zigbee/examples/Zigbee_Vibration_Sensor/Zigbee_Vibration_Sensor.ino + :language: arduino diff --git a/docs/en/zigbee/ep_wind_speed_sensor.rst b/docs/en/zigbee/ep_wind_speed_sensor.rst new file mode 100644 index 00000000000..67c4958e37c --- /dev/null +++ b/docs/en/zigbee/ep_wind_speed_sensor.rst @@ -0,0 +1,119 @@ +##################### +ZigbeeWindSpeedSensor +##################### + +About +----- + +The ``ZigbeeWindSpeedSensor`` class provides a wind speed sensor endpoint for Zigbee networks. This endpoint implements the Zigbee Home Automation (HA) standard for wind speed measurement devices. + +**Features:** +* Wind speed measurement in m/s +* Configurable measurement range +* Tolerance and reporting configuration +* Automatic reporting capabilities +* Integration with common endpoint features (binding, OTA, etc.) +* Zigbee HA standard compliance + +**Use Cases:** +* Weather stations +* Wind turbine monitoring +* Agricultural weather monitoring +* Marine applications +* Smart home weather systems +* Industrial wind monitoring + +API Reference +------------- + +Constructor +*********** + +ZigbeeWindSpeedSensor +^^^^^^^^^^^^^^^^^^^^^ + +Creates a new Zigbee wind speed sensor endpoint. + +.. code-block:: arduino + + ZigbeeWindSpeedSensor(uint8_t endpoint); + +* ``endpoint`` - Endpoint number (1-254) + +API Methods +*********** + +setWindSpeed +^^^^^^^^^^^^ + +Sets the wind speed measurement value. + +.. code-block:: arduino + + bool setWindSpeed(float value); + +* ``value`` - Wind speed value in 0.01 m/s + +This function will return ``true`` if successful, ``false`` otherwise. + +setMinMaxValue +^^^^^^^^^^^^^^ + +Sets the minimum and maximum measurement values. + +.. code-block:: arduino + + bool setMinMaxValue(float min, float max); + +* ``min`` - Minimum wind speed value in 0.01 m/s +* ``max`` - Maximum wind speed value in 0.01 m/s + +This function will return ``true`` if successful, ``false`` otherwise. + +setTolerance +^^^^^^^^^^^^ + +Sets the tolerance value for measurements. + +.. code-block:: arduino + + bool setTolerance(float tolerance); + +* ``tolerance`` - Tolerance value in 0.01 m/s + +This function will return ``true`` if successful, ``false`` otherwise. + +setReporting +^^^^^^^^^^^^ + +Sets the reporting configuration for wind speed measurements. + +.. code-block:: arduino + + bool setReporting(uint16_t min_interval, uint16_t max_interval, float delta); + +* ``min_interval`` - Minimum reporting interval in seconds +* ``max_interval`` - Maximum reporting interval in seconds +* ``delta`` - Minimum change required to trigger a report in 0.01 m/s + +This function will return ``true`` if successful, ``false`` otherwise. + +reportWindSpeed +^^^^^^^^^^^^^^^ + +Manually reports the current wind speed value. + +.. code-block:: arduino + + bool reportWindSpeed(); + +This function will return ``true`` if successful, ``false`` otherwise. + +Example +------- + +Wind Speed Sensor Implementation +******************************** + +.. literalinclude:: ../../../libraries/Zigbee/examples/Zigbee_Wind_Speed_Sensor/Zigbee_Wind_Speed_Sensor.ino + :language: arduino diff --git a/docs/en/zigbee/ep_window_covering.rst b/docs/en/zigbee/ep_window_covering.rst new file mode 100644 index 00000000000..a4b8cd918a6 --- /dev/null +++ b/docs/en/zigbee/ep_window_covering.rst @@ -0,0 +1,233 @@ +#################### +ZigbeeWindowCovering +#################### + +About +----- + +The ``ZigbeeWindowCovering`` class provides a window covering endpoint for Zigbee networks. This endpoint implements the Zigbee Home Automation (HA) standard for motorized blinds, shades, and other window coverings. + +**Features:** +* Position control (lift and tilt) +* Multiple window covering types support +* Configurable operation modes and limits +* Status reporting and callbacks +* Safety features and limits + +**Supported Window Covering Types:** +* ROLLERSHADE - Lift support +* ROLLERSHADE_2_MOTOR - Lift support +* ROLLERSHADE_EXTERIOR - Lift support +* ROLLERSHADE_EXTERIOR_2_MOTOR - Lift support +* DRAPERY - Lift support +* AWNING - Lift support +* SHUTTER - Tilt support +* BLIND_TILT_ONLY - Tilt support +* BLIND_LIFT_AND_TILT - Lift and Tilt support +* PROJECTOR_SCREEN - Lift support + +API Reference +------------- + +Constructor +*********** + +ZigbeeWindowCovering +^^^^^^^^^^^^^^^^^^^^ + +Creates a new Zigbee window covering endpoint. + +.. code-block:: arduino + + ZigbeeWindowCovering(uint8_t endpoint); + +* ``endpoint`` - Endpoint number (1-254) + +Position Control +**************** + +setLiftPosition +^^^^^^^^^^^^^^^ + +Sets the window covering lift position. + +.. code-block:: arduino + + bool setLiftPosition(uint16_t lift_position); + +* ``lift_position`` - Lift position + +This function will return ``true`` if successful, ``false`` otherwise. + +setLiftPercentage +^^^^^^^^^^^^^^^^^ + +Sets the window covering lift position as a percentage. + +.. code-block:: arduino + + bool setLiftPercentage(uint8_t lift_percentage); + +* ``lift_percentage`` - Lift percentage (0-100, where 0 is fully closed, 100 is fully open) + +This function will return ``true`` if successful, ``false`` otherwise. + +setTiltPosition +^^^^^^^^^^^^^^^ + +Sets the window covering tilt position in degrees. + +.. code-block:: arduino + + bool setTiltPosition(uint16_t tilt_position); + +* ``tilt_position`` - Tilt position in degrees + +This function will return ``true`` if successful, ``false`` otherwise. + +setTiltPercentage +^^^^^^^^^^^^^^^^^ + +Sets the window covering tilt position as a percentage. + +.. code-block:: arduino + + bool setTiltPercentage(uint8_t tilt_percentage); + +* ``tilt_percentage`` - Tilt percentage (0-100) + +This function will return ``true`` if successful, ``false`` otherwise. + +Configuration +************* + +setCoveringType +^^^^^^^^^^^^^^^ + +Sets the window covering type. + +.. code-block:: arduino + + bool setCoveringType(ZigbeeWindowCoveringType covering_type); + +* ``covering_type`` - Window covering type (see supported types above) + +This function will return ``true`` if successful, ``false`` otherwise. + +setConfigStatus +^^^^^^^^^^^^^^^ + +Sets the window covering configuration status. + +.. code-block:: arduino + + bool setConfigStatus(bool operational, bool online, bool commands_reversed, bool lift_closed_loop, bool tilt_closed_loop, bool lift_encoder_controlled, bool tilt_encoder_controlled); + +* ``operational`` - Operational status +* ``online`` - Online status +* ``commands_reversed`` - Commands reversed flag +* ``lift_closed_loop`` - Lift closed loop flag +* ``tilt_closed_loop`` - Tilt closed loop flag +* ``lift_encoder_controlled`` - Lift encoder controlled flag +* ``tilt_encoder_controlled`` - Tilt encoder controlled flag + +This function will return ``true`` if successful, ``false`` otherwise. + +setMode +^^^^^^^ + +Sets the window covering operation mode. + +.. code-block:: arduino + + bool setMode(bool motor_reversed, bool calibration_mode, bool maintenance_mode, bool leds_on); + +* ``motor_reversed`` - Motor reversed flag +* ``calibration_mode`` - Calibration mode flag +* ``maintenance_mode`` - Maintenance mode flag +* ``leds_on`` - LEDs on flag + +This function will return ``true`` if successful, ``false`` otherwise. + +setLimits +^^^^^^^^^ + +Sets the motion limits for the window covering. + +.. code-block:: arduino + + bool setLimits(uint16_t installed_open_limit_lift, uint16_t installed_closed_limit_lift, uint16_t installed_open_limit_tilt, uint16_t installed_closed_limit_tilt); + +* ``installed_open_limit_lift`` - Installed open limit for lift +* ``installed_closed_limit_lift`` - Installed closed limit for lift +* ``installed_open_limit_tilt`` - Installed open limit for tilt +* ``installed_closed_limit_tilt`` - Installed closed limit for tilt + +This function will return ``true`` if successful, ``false`` otherwise. + +Event Handling +************** + +onOpen +^^^^^^ + +Sets a callback function to be called when the window covering opens. + +.. code-block:: arduino + + void onOpen(void (*callback)()); + +* ``callback`` - Function to call when window covering opens + +onClose +^^^^^^^ + +Sets a callback function to be called when the window covering closes. + +.. code-block:: arduino + + void onClose(void (*callback)()); + +* ``callback`` - Function to call when window covering closes + +onGoToLiftPercentage +^^^^^^^^^^^^^^^^^^^^ + +Sets a callback function to be called when lift percentage changes. + +.. code-block:: arduino + + void onGoToLiftPercentage(void (*callback)(uint8_t)); + +* ``callback`` - Function to call when lift percentage changes + +onGoToTiltPercentage +^^^^^^^^^^^^^^^^^^^^ + +Sets a callback function to be called when tilt percentage changes. + +.. code-block:: arduino + + void onGoToTiltPercentage(void (*callback)(uint8_t)); + +* ``callback`` - Function to call when tilt percentage changes + +onStop +^^^^^^ + +Sets a callback function to be called when window covering stops. + +.. code-block:: arduino + + void onStop(void (*callback)()); + +* ``callback`` - Function to call when window covering stops + +Example +------- + +Window Covering Implementation +****************************** + +.. literalinclude:: ../../../libraries/Zigbee/examples/Zigbee_Window_Covering/Zigbee_Window_Covering.ino + :language: arduino diff --git a/docs/en/zigbee/zigbee.rst b/docs/en/zigbee/zigbee.rst new file mode 100644 index 00000000000..ba14adad8e1 --- /dev/null +++ b/docs/en/zigbee/zigbee.rst @@ -0,0 +1,159 @@ +###### +Zigbee +###### + +About +----- + +The Zigbee library provides support for creating Zigbee 3.0 compatible devices including: + +* Support for different Zigbee roles (Coordinator, Router, End Device) +* Network management (scanning, joining, commissioning) +* Multiple endpoint types for various device categories +* OTA (Over-The-Air) update support +* Power management for battery-powered devices +* Time synchronization +* Advanced binding and group management + +The Zigbee library is built on top of `ESP-ZIGBEE-SDK `_ and provides a high-level Arduino-style interface for creating Zigbee devices. + +Zigbee Network Topology +*********************** + +.. code-block:: text + + ┌─────────────────┐ ┌─────────────────┐ ┌─────────────────┐ + │ Coordinator │◄─────►│ Router │◄─────►│ Router │ + │ (Gateway) │ │ (Repeater) │ │ (Thermostat) │ + └─────────────────┘ └─────────────────┘ └─────────────────┘ + │ │ │ + │ ▼ │ + │ ┌─────────────────┐ │ + │ │ End Device │ │ + │ │ (Sensor) │ │ + │ └─────────────────┘ │ + │ │ + ▼ ▼ + ┌─────────────────┐ ┌─────────────────┐ + │ End Device │ │ End Device │ + │ (Sensor) │ │ (Sensor) │ + └─────────────────┘ └─────────────────┘ + + +**Device Roles** + +* **Coordinator**: Forms and manages the network, stores network information +* **Router**: Extends network range, routes messages, mains powered devices (typically lights, switches, etc.) +* **End Device**: Battery-powered devices that can sleep for extended periods (typically sensors) + +Zigbee Library Structure +------------------------ + +**The library is split into three main components:** + +* ``ZigbeeCore``: The main class that manages the Zigbee network +* ``ZigbeeEP``: The base class for all Zigbee endpoints, which provides common functionality for all endpoint types +* ``Specific endpoint classes``: The classes for all Zigbee endpoints, which provides the specific functionality for each endpoint type + +ZigbeeCore +********** + +The ``ZigbeeCore`` class is the main entry point for all Zigbee operations. It serves as the central coordinator that manages: + +* **Network Operations**: Starting, stopping, and managing the Zigbee network +* **Device Role Management**: Configuring the device as Coordinator, Router, or End Device +* **Endpoint Management**: Adding and managing multiple device endpoints +* **Network Discovery**: Scanning for and joining existing networks +* **OTA Updates**: Managing over-the-air firmware updates +* **Power Management**: Configuring sleep modes for battery-powered devices + +The ``ZigbeeCore`` class is implemented as a singleton, meaning there's only one instance available globally. You access it directly as ``Zigbee`` without creating an instance. + +.. toctree:: + :maxdepth: 3 + + zigbee_core + +ZigbeeEP +******** + +The ``ZigbeeEP`` class is the base class for all Zigbee endpoints. It provides common functionality for all endpoint types. + +* **Device Information**: Every endpoint can be configured with manufacturer and model information that helps identify the device on the network +* **Binding Management**: Binding allows endpoints to establish direct communication links with other devices. This enables automatic command transmission without requiring manual addressing +* **Power Management**: Endpoints can report their power source type and battery status, which helps the network optimize communication patterns +* **Time Synchronization**: Endpoints can synchronize with network time, enabling time-based operations and scheduling +* **OTA Support**: Endpoints can receive over-the-air firmware updates to add new features or fix bugs +* **Device Discovery**: Endpoints can read manufacturer and model information from other devices on the network + +.. toctree:: + :maxdepth: 2 + + zigbee_ep + +Specific endpoint classes +************************* + +Library provides the following endpoint classes from lights, switches, sensors, etc. Each endpoint class provides the specific functionality for each endpoint type and inherits from the ``ZigbeeEP`` class. + +.. toctree:: + :maxdepth: 1 + :glob: + + ep_* + + +Common Problems and Issues +-------------------------- + +Troubleshooting +--------------- + +Common Issues +************* + +**Device won't join network** + * Ensure the coordinator is in pairing mode + * Check that the device is configured with the correct role + * Verify the channel mask includes the coordinator's channel + +**OTA updates fail** + * Ensure the OTA server is properly configured + * Check that the device has sufficient memory for the update + * Verify network connectivity + +**Battery devices not working** + * Ensure proper power source configuration + * Check sleep/wake timing settings + * Verify parent device (router/coordinator) is always powered + +**Binding issues** + * Check that both devices support the required clusters + * Verify that binding is enabled on both devices + * Ensure devices are on the same network + +**Network connectivity problems** + * Check that devices are within range + * Verify that routers are properly configured + * Check for interference from other 2.4 GHz devices + +Factory Reset +************* + +If you have problems with connecting to the network, you can try to factory reset the device. This will erase all the network settings and act as brand new device. + +.. code-block:: arduino + + Zigbee.factoryReset(true); // true = restart after reset + +Debug Mode +********** + +For better debugging, you can enable debug mode to get detailed information about network operations. Call debug mode before starting Zigbee. +Also selecting zigbee mode with *debug* suffix is recommended. + +.. code-block:: arduino + + Zigbee.setDebugMode(true); + // Start Zigbee with debug output + Zigbee.begin(); diff --git a/docs/en/zigbee/zigbee_core.rst b/docs/en/zigbee/zigbee_core.rst new file mode 100644 index 00000000000..ee8f7234210 --- /dev/null +++ b/docs/en/zigbee/zigbee_core.rst @@ -0,0 +1,375 @@ +########## +ZigbeeCore +########## + +About +----- + +The ``ZigbeeCore`` class is the main entry point for all Zigbee operations. It serves as the central class that manages: + +* **Network Operations**: Starting, stopping, and managing the Zigbee network +* **Device Role Management**: Configuring the device as Coordinator, Router, or End Device +* **Endpoint Management**: Adding and managing multiple device endpoints +* **Network Discovery**: Scanning for and joining existing networks + +ZigbeeCore APIs +--------------- + +Network Initialization +********************** + +begin +^^^^^ + +Initializes the Zigbee stack and starts the network. + +.. code-block:: arduino + + bool begin(zigbee_role_t role = ZIGBEE_END_DEVICE, bool erase_nvs = false); + bool begin(esp_zb_cfg_t *role_cfg, bool erase_nvs = false); + +* ``role`` - Device role (default: ``ZIGBEE_END_DEVICE``) +* ``role_cfg`` - Custom role configuration structure +* ``erase_nvs`` - Whether to erase NVS storage (default: ``false``) + +This function will return ``true`` if initialization successful, ``false`` otherwise. + +**Available Roles:** + +* **ZIGBEE_COORDINATOR**: Network coordinator, forms and manages the network +* **ZIGBEE_ROUTER**: Network router, connects to existing network and extends network range and routes messages (if device is mains powered, always use this role) +* **ZIGBEE_END_DEVICE**: End device, connects to existing network (typically battery-powered which can sleep) + +.. note:: + + Depending on the Zigbee role, proper Zigbee mode and partition scheme must be set in the Arduino IDE. + + * **ZIGBEE_COORDINATOR** and **ZIGBEE_ROUTER**: + * Zigbee mode to ``Zigbee ZCZR (coordinator/router)``. + * Partition scheme to ``Zigbee ZCZR xMB with spiffs`` (where ``x`` is the number of MB of selected flash size). + * **ZIGBEE_END_DEVICE**: + * Zigbee mode to ``Zigbee ED (end device)``. + * Partition scheme to ``Zigbee xMB with spiffs`` (where ``x`` is the number of MB of selected flash size). + +Network Status +************** + +started +^^^^^^^ + +Checks if the Zigbee stack has been started. + +.. code-block:: arduino + + bool started(); + +This function will return ``true`` if Zigbee stack is running, ``false`` otherwise. + +connected +^^^^^^^^^ + +Checks if the device is connected to a Zigbee network. + +.. code-block:: arduino + + bool connected(); + +This function will return ``true`` if connected to network, ``false`` otherwise. + +getRole +^^^^^^^ + +Gets the current Zigbee device role. + +.. code-block:: arduino + + zigbee_role_t getRole(); + +This function will return current device role (``ZIGBEE_COORDINATOR``, ``ZIGBEE_ROUTER``, ``ZIGBEE_END_DEVICE``). + +Endpoint Management +******************* + +addEndpoint +^^^^^^^^^^^ + +Adds an endpoint to the Zigbee network. + +.. code-block:: arduino + + bool addEndpoint(ZigbeeEP *ep); + +* ``ep`` - Pointer to the endpoint object to add + +This function will return ``true`` if endpoint added successfully, ``false`` otherwise. + +Network Configuration +********************* + +setPrimaryChannelMask +^^^^^^^^^^^^^^^^^^^^^ + +Sets the primary channel mask for network scanning and joining. + +.. code-block:: arduino + + void setPrimaryChannelMask(uint32_t mask); + +* ``mask`` - Channel mask (default: all channels 11-26, mask 0x07FFF800) + +setScanDuration +^^^^^^^^^^^^^^^ + +Sets the scan duration for network discovery. + +.. code-block:: arduino + + void setScanDuration(uint8_t duration); + +* ``duration`` - Scan duration (1-4, where 1 is fastest, 4 is slowest) + +getScanDuration +^^^^^^^^^^^^^^^ + +Gets the current scan duration setting. + +.. code-block:: arduino + + uint8_t getScanDuration(); + +This function will return current scan duration (1-4). + +Power Management +**************** + +setRxOnWhenIdle +^^^^^^^^^^^^^^^ + +Sets whether the device keeps its receiver on when idle. + +.. code-block:: arduino + + void setRxOnWhenIdle(bool rx_on_when_idle); + +* ``rx_on_when_idle`` - ``true`` to keep receiver on, ``false`` to allow sleep + +getRxOnWhenIdle +^^^^^^^^^^^^^^^ + +Gets the current receiver idle setting. + +.. code-block:: arduino + + bool getRxOnWhenIdle(); + +This function will return current receiver idle setting. + +setTimeout +^^^^^^^^^^ + +Sets the timeout for network operations. + +.. code-block:: arduino + + void setTimeout(uint32_t timeout); + +* ``timeout`` - Timeout in milliseconds (default: 30000 ms) + +Network Discovery +***************** + +scanNetworks +^^^^^^^^^^^^ + +Scans for available Zigbee networks. + +.. code-block:: arduino + + void scanNetworks(uint32_t channel_mask = ESP_ZB_TRANSCEIVER_ALL_CHANNELS_MASK, uint8_t scan_duration = 5); + +* ``channel_mask`` - Channels to scan (default: all channels) +* ``scan_duration`` - Scan duration (default: 5) + +scanComplete +^^^^^^^^^^^^ + +Checks if network scanning is complete. + +.. code-block:: arduino + + int16_t scanComplete(); + +This function will return: +* ``-2``: Scan failed or not started +* ``-1``: Scan running +* ``0``: No networks found +* ``>0``: Number of networks found + +getScanResult +^^^^^^^^^^^^^ + +Gets the scan results. + +.. code-block:: arduino + + zigbee_scan_result_t *getScanResult(); + +This function will return pointer to scan results, or ``NULL`` if no results. + +scanDelete +^^^^^^^^^^ + +Deletes the scan results from memory. + +.. code-block:: arduino + + void scanDelete(); + +Network Management (Coordinator only) +************************************* + +setRebootOpenNetwork +^^^^^^^^^^^^^^^^^^^^ + +Opens the network for joining after reboot for a specified time. + +.. code-block:: arduino + + void setRebootOpenNetwork(uint8_t time); + +* ``time`` - Time in seconds to keep network open after reboot + +openNetwork +^^^^^^^^^^^ + +Opens the network for device joining for a specified time. + +.. code-block:: arduino + + void openNetwork(uint8_t time); + +* ``time`` - Time in seconds to keep network open for device joining + +closeNetwork +^^^^^^^^^^^^ + +Closes the network to prevent new devices from joining. + +.. code-block:: arduino + + void closeNetwork(); + +Radio Configuration +******************* + +setRadioConfig +^^^^^^^^^^^^^^ + +Sets the radio configuration. + +.. code-block:: arduino + + void setRadioConfig(esp_zb_radio_config_t config); + +* ``config`` - Radio configuration structure + +getRadioConfig +^^^^^^^^^^^^^^ + +Gets the current radio configuration. + +.. code-block:: arduino + + esp_zb_radio_config_t getRadioConfig(); + +This function will return current radio configuration. + +Host Configuration +****************** + +setHostConfig +^^^^^^^^^^^^^ + +Sets the host configuration. + +.. code-block:: arduino + + void setHostConfig(esp_zb_host_config_t config); + +* ``config`` - Host configuration structure + +getHostConfig +^^^^^^^^^^^^^ + +Gets the current host configuration. + +.. code-block:: arduino + + esp_zb_host_config_t getHostConfig(); + +This function will return current host configuration. + +Debug and Utilities +******************* + +setDebugMode +^^^^^^^^^^^^ + +Enables or disables debug mode. + +.. code-block:: arduino + + void setDebugMode(bool debug); + +* ``debug`` - ``true`` to enable debug output, ``false`` to disable + +getDebugMode +^^^^^^^^^^^^ + +Gets the current debug mode setting. + +.. code-block:: arduino + + bool getDebugMode(); + +This function will return current debug mode setting. + +factoryReset +^^^^^^^^^^^^ + +Performs a factory reset, clearing all network settings. + +.. code-block:: arduino + + void factoryReset(bool restart = true); + +* ``restart`` - ``true`` to restart after reset (default: ``true``) + +Utility Functions +***************** + +formatIEEEAddress +^^^^^^^^^^^^^^^^^ + +Formats an IEEE address for display. + +.. code-block:: arduino + + static const char *formatIEEEAddress(const esp_zb_ieee_addr_t addr); + +* ``addr`` - IEEE address to format + +This function will return formatted address string. + +formatShortAddress +^^^^^^^^^^^^^^^^^^ + +Formats a short address for display. + +.. code-block:: arduino + + static const char *formatShortAddress(uint16_t addr); + +* ``addr`` - Short address to format + +This function will return formatted address string. diff --git a/docs/en/zigbee/zigbee_ep.rst b/docs/en/zigbee/zigbee_ep.rst new file mode 100644 index 00000000000..ce13d60bbaf --- /dev/null +++ b/docs/en/zigbee/zigbee_ep.rst @@ -0,0 +1,357 @@ +######## +ZigbeeEP +######## + +About +----- + +The ``ZigbeeEP`` class is the base class for all Zigbee endpoints. It provides common functionality for all endpoint types. + +* **Device Information**: Every endpoint can be configured with manufacturer and model information that helps identify the device on the network +* **Binding Management**: Binding allows endpoints to establish direct communication links with other devices. This enables automatic command transmission without requiring manual addressing +* **Power Management**: Endpoints can report their power source type and battery status, which helps the network optimize communication patterns +* **Time Synchronization**: Endpoints can synchronize with network time, enabling time-based operations and scheduling +* **OTA Support**: Endpoints can receive over-the-air firmware updates to add new features or fix bugs +* **Device Discovery**: Endpoints can read manufacturer and model information from other devices on the network + + +ZigbeeEP APIs +------------- + +Device Information +****************** + +setManufacturerAndModel +^^^^^^^^^^^^^^^^^^^^^^^ + +Sets the manufacturer name and model identifier for the device. + +.. code-block:: arduino + + bool setManufacturerAndModel(const char *name, const char *model); + +* ``name`` - Manufacturer name (max 32 characters) +* ``model`` - Model identifier (max 32 characters) + +This function will return ``true`` if set successfully, ``false`` otherwise. + +getEndpoint +^^^^^^^^^^^ + +Gets the endpoint number assigned to this device. + +.. code-block:: arduino + + uint8_t getEndpoint(); + +This function will return the endpoint number (1-254). + +Binding Management +****************** + +bound +^^^^^ + +Checks if the endpoint has any bound devices. + +.. code-block:: arduino + + bool bound(); + +This function will return ``true`` if the endpoint has bound devices, ``false`` otherwise. + +getBoundDevices +^^^^^^^^^^^^^^^ + +Gets the list of devices bound to this endpoint. + +.. code-block:: arduino + + std::vector getBoundDevices(); + +This function will return list of bound device parameters. + +printBoundDevices +^^^^^^^^^^^^^^^^^ + +Prints information about bound devices to Serial or a custom Print object. + +.. code-block:: arduino + + void printBoundDevices(Print &print = Serial); + +* ``print`` - Custom Print object (optional, defaults to Serial) + +allowMultipleBinding +^^^^^^^^^^^^^^^^^^^^ + +Enables or disables multiple device binding for this endpoint. + +.. code-block:: arduino + + void allowMultipleBinding(bool bind); + +* ``bind`` - ``true`` to allow multiple bindings, ``false`` for single binding only + +setManualBinding +^^^^^^^^^^^^^^^^ + +Enables or disables manual binding mode. Manual binding mode is supposed to be used when using ZHA or Z2M where you bind devices manually. + +.. code-block:: arduino + + void setManualBinding(bool bind); + +* ``bind`` - ``true`` for manual binding, ``false`` for automatic binding (default) + +clearBoundDevices +^^^^^^^^^^^^^^^^^ + +Removes all bound devices from this endpoint. + +.. code-block:: arduino + + void clearBoundDevices(); + +Binding Status +************** + +epAllowMultipleBinding +^^^^^^^^^^^^^^^^^^^^^^ + +Gets whether multiple device binding is allowed for this endpoint. + +.. code-block:: arduino + + bool epAllowMultipleBinding(); + +This function will return ``true`` if multiple bindings are allowed, ``false`` otherwise. + +epUseManualBinding +^^^^^^^^^^^^^^^^^^ + +Gets whether manual binding mode is enabled for this endpoint. + +.. code-block:: arduino + + bool epUseManualBinding(); + +This function will return ``true`` if manual binding is enabled, ``false`` otherwise. + +Power Management +**************** + +setPowerSource +^^^^^^^^^^^^^^ + +Sets the power source type for the endpoint. + +.. code-block:: arduino + + bool setPowerSource(uint8_t source, uint8_t percentage = 0xff, uint8_t voltage = 0xff); + +* ``source`` - Power source type (``ZB_POWER_SOURCE_MAINS``, ``ZB_POWER_SOURCE_BATTERY``, etc.) +* ``percentage`` - Battery percentage (0-100, default: 0xff) +* ``voltage`` - Battery voltage in 100 mV units (default: 0xff) + +This function will return ``true`` if set successfully, ``false`` otherwise. + +setBatteryPercentage +^^^^^^^^^^^^^^^^^^^^ + +Sets the current battery percentage. + +.. code-block:: arduino + + bool setBatteryPercentage(uint8_t percentage); + +* ``percentage`` - Battery percentage (0-100) + +This function will return ``true`` if set successfully, ``false`` otherwise. + +setBatteryVoltage +^^^^^^^^^^^^^^^^^ + +Sets the battery voltage. + +.. code-block:: arduino + + bool setBatteryVoltage(uint8_t voltage); + +* ``voltage`` - Battery voltage in 100 mV units (e.g., 35 for 3.5 V) + +This function will return ``true`` if set successfully, ``false`` otherwise. + +reportBatteryPercentage +^^^^^^^^^^^^^^^^^^^^^^^ + +Reports the current battery percentage to the network. + +.. code-block:: arduino + + bool reportBatteryPercentage(); + +This function will return ``true`` if reported successfully, ``false`` otherwise. + +Time Synchronization +******************** + +addTimeCluster +^^^^^^^^^^^^^^ + +Adds time synchronization cluster to the endpoint. When you want to add a server cluster (have the time and GMT offset) fill the time structure with the current time and GMT offset. +For client cluster (get the time and GMT offset) keep the default parameters. + +.. code-block:: arduino + + bool addTimeCluster(tm time = {}, int32_t gmt_offset = 0); + +* ``time`` - Current time structure (default: empty) +* ``gmt_offset`` - GMT offset in seconds (default: 0) + +This function will return ``true`` if added successfully, ``false`` otherwise. + +setTime +^^^^^^^ + +Sets the current time for the endpoint. + +.. code-block:: arduino + + bool setTime(tm time); + +* ``time`` - Time structure to set + +This function will return ``true`` if set successfully, ``false`` otherwise. + +setTimezone +^^^^^^^^^^^ + +Sets the timezone offset for the endpoint. + +.. code-block:: arduino + + bool setTimezone(int32_t gmt_offset); + +* ``gmt_offset`` - GMT offset in seconds + +This function will return ``true`` if set successfully, ``false`` otherwise. + +getTime +^^^^^^^ + +Gets the current network time. + +.. code-block:: arduino + + struct tm getTime(uint8_t endpoint = 1, int32_t short_addr = 0x0000, esp_zb_ieee_addr_t ieee_addr = {}); + +* ``endpoint`` - Target endpoint (default: 1) +* ``short_addr`` - Target device short address (default: 0x0000) +* ``ieee_addr`` - Target device IEEE address (default: empty) + +This function will return network time structure. + +getTimezone +^^^^^^^^^^^ + +Gets the timezone offset. + +.. code-block:: arduino + + int32_t getTimezone(uint8_t endpoint = 1, int32_t short_addr = 0x0000, esp_zb_ieee_addr_t ieee_addr = {}); + +* ``endpoint`` - Target endpoint (default: 1) +* ``short_addr`` - Target device short address (default: 0x0000) +* ``ieee_addr`` - Target device IEEE address (default: empty) + +This function will return GMT offset in seconds. + +OTA Support +*********** + +addOTAClient +^^^^^^^^^^^^ + +Adds OTA client to the endpoint for firmware updates. + +.. code-block:: arduino + + bool 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); + +* ``file_version`` - Current firmware version +* ``downloaded_file_ver`` - Downloaded file version +* ``hw_version`` - Hardware version +* ``manufacturer`` - Manufacturer code (default: 0x1001) +* ``image_type`` - Image type code (default: 0x1011) +* ``max_data_size`` - Maximum data size for OTA transfer (default: 223) + +This function will return ``true`` if added successfully, ``false`` otherwise. + +requestOTAUpdate +^^^^^^^^^^^^^^^^ + +Requests OTA update from the server. + +.. code-block:: arduino + + void requestOTAUpdate(); + +Device Discovery +**************** + +readManufacturer +^^^^^^^^^^^^^^^^ + +Reads the manufacturer name from a remote device. + +.. code-block:: arduino + + char *readManufacturer(uint8_t endpoint, uint16_t short_addr, esp_zb_ieee_addr_t ieee_addr); + +* ``endpoint`` - Target endpoint number +* ``short_addr`` - Target device short address +* ``ieee_addr`` - Target device IEEE address + +This function will return pointer to manufacturer string, or ``NULL`` if read failed. + +readModel +^^^^^^^^^ + +Reads the model identifier from a remote device. + +.. code-block:: arduino + + char *readModel(uint8_t endpoint, uint16_t short_addr, esp_zb_ieee_addr_t ieee_addr); + +* ``endpoint`` - Target endpoint number +* ``short_addr`` - Target device short address +* ``ieee_addr`` - Target device IEEE address + +This function will return pointer to model string, or ``NULL`` if read failed. + +Event Handling +************** + +onIdentify +^^^^^^^^^^ + +Sets a callback function for identify events. + +.. code-block:: arduino + + void onIdentify(void (*callback)(uint16_t)); + +* ``callback`` - Function to call when identify event occurs +* ``time`` - Identify time in seconds + +Supported Endpoints +------------------- + +The Zigbee library provides specialized endpoint classes for different device types. Each endpoint type includes specific clusters and functionality relevant to that device category. + +.. toctree:: + :maxdepth: 1 + :glob: + + ep_* From 554de56f40e68d591b945bd1a8c931798e8a9167 Mon Sep 17 00:00:00 2001 From: Me No Dev Date: Tue, 22 Jul 2025 11:23:49 +0300 Subject: [PATCH 136/173] IDF release/v5.5 25c7c119 (#11623) --- package/package_esp32_index.template.json | 68 +++++++++++------------ 1 file changed, 34 insertions(+), 34 deletions(-) diff --git a/package/package_esp32_index.template.json b/package/package_esp32_index.template.json index 6e2dd0ab337..f844a91161a 100644 --- a/package/package_esp32_index.template.json +++ b/package/package_esp32_index.template.json @@ -51,7 +51,7 @@ { "packager": "esp32", "name": "esp32-arduino-libs", - "version": "idf-release_v5.5-cf8dad07-v1" + "version": "idf-release_v5.5-25c7c119-v1" }, { "packager": "esp32", @@ -104,63 +104,63 @@ "tools": [ { "name": "esp32-arduino-libs", - "version": "idf-release_v5.5-cf8dad07-v1", + "version": "idf-release_v5.5-25c7c119-v1", "systems": [ { "host": "i686-mingw32", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-cf8dad07-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-cf8dad07-v1.zip", - "checksum": "SHA-256:c8310c661871a5d82b8b09a2b8d792936bf9a5023ed6dcb683ff93ff9ea4aaa7", - "size": "430423461" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-25c7c119-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-25c7c119-v1.zip", + "checksum": "SHA-256:5760b8d2215c07b0f054e9cd62a268f218c2026db8b395e90696d8038f69d9a8", + "size": "430472412" }, { "host": "x86_64-mingw32", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-cf8dad07-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-cf8dad07-v1.zip", - "checksum": "SHA-256:c8310c661871a5d82b8b09a2b8d792936bf9a5023ed6dcb683ff93ff9ea4aaa7", - "size": "430423461" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-25c7c119-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-25c7c119-v1.zip", + "checksum": "SHA-256:5760b8d2215c07b0f054e9cd62a268f218c2026db8b395e90696d8038f69d9a8", + "size": "430472412" }, { "host": "arm64-apple-darwin", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-cf8dad07-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-cf8dad07-v1.zip", - "checksum": "SHA-256:c8310c661871a5d82b8b09a2b8d792936bf9a5023ed6dcb683ff93ff9ea4aaa7", - "size": "430423461" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-25c7c119-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-25c7c119-v1.zip", + "checksum": "SHA-256:5760b8d2215c07b0f054e9cd62a268f218c2026db8b395e90696d8038f69d9a8", + "size": "430472412" }, { "host": "x86_64-apple-darwin", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-cf8dad07-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-cf8dad07-v1.zip", - "checksum": "SHA-256:c8310c661871a5d82b8b09a2b8d792936bf9a5023ed6dcb683ff93ff9ea4aaa7", - "size": "430423461" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-25c7c119-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-25c7c119-v1.zip", + "checksum": "SHA-256:5760b8d2215c07b0f054e9cd62a268f218c2026db8b395e90696d8038f69d9a8", + "size": "430472412" }, { "host": "x86_64-pc-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-cf8dad07-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-cf8dad07-v1.zip", - "checksum": "SHA-256:c8310c661871a5d82b8b09a2b8d792936bf9a5023ed6dcb683ff93ff9ea4aaa7", - "size": "430423461" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-25c7c119-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-25c7c119-v1.zip", + "checksum": "SHA-256:5760b8d2215c07b0f054e9cd62a268f218c2026db8b395e90696d8038f69d9a8", + "size": "430472412" }, { "host": "i686-pc-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-cf8dad07-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-cf8dad07-v1.zip", - "checksum": "SHA-256:c8310c661871a5d82b8b09a2b8d792936bf9a5023ed6dcb683ff93ff9ea4aaa7", - "size": "430423461" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-25c7c119-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-25c7c119-v1.zip", + "checksum": "SHA-256:5760b8d2215c07b0f054e9cd62a268f218c2026db8b395e90696d8038f69d9a8", + "size": "430472412" }, { "host": "aarch64-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-cf8dad07-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-cf8dad07-v1.zip", - "checksum": "SHA-256:c8310c661871a5d82b8b09a2b8d792936bf9a5023ed6dcb683ff93ff9ea4aaa7", - "size": "430423461" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-25c7c119-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-25c7c119-v1.zip", + "checksum": "SHA-256:5760b8d2215c07b0f054e9cd62a268f218c2026db8b395e90696d8038f69d9a8", + "size": "430472412" }, { "host": "arm-linux-gnueabihf", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-cf8dad07-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-cf8dad07-v1.zip", - "checksum": "SHA-256:c8310c661871a5d82b8b09a2b8d792936bf9a5023ed6dcb683ff93ff9ea4aaa7", - "size": "430423461" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-25c7c119-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-25c7c119-v1.zip", + "checksum": "SHA-256:5760b8d2215c07b0f054e9cd62a268f218c2026db8b395e90696d8038f69d9a8", + "size": "430472412" } ] }, From 69d891434bd5cd327cd93516fc60941f3e2bfd85 Mon Sep 17 00:00:00 2001 From: Me No Dev Date: Wed, 23 Jul 2025 01:37:23 +0300 Subject: [PATCH 137/173] IDF release/v5.5 b66b5448 (#11626) IDF release/v5.5 b66b5448 --- package/package_esp32_index.template.json | 68 +++++++++++------------ 1 file changed, 34 insertions(+), 34 deletions(-) diff --git a/package/package_esp32_index.template.json b/package/package_esp32_index.template.json index f844a91161a..3909c833c56 100644 --- a/package/package_esp32_index.template.json +++ b/package/package_esp32_index.template.json @@ -51,7 +51,7 @@ { "packager": "esp32", "name": "esp32-arduino-libs", - "version": "idf-release_v5.5-25c7c119-v1" + "version": "idf-release_v5.5-b66b5448-v1" }, { "packager": "esp32", @@ -104,63 +104,63 @@ "tools": [ { "name": "esp32-arduino-libs", - "version": "idf-release_v5.5-25c7c119-v1", + "version": "idf-release_v5.5-b66b5448-v1", "systems": [ { "host": "i686-mingw32", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-25c7c119-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-25c7c119-v1.zip", - "checksum": "SHA-256:5760b8d2215c07b0f054e9cd62a268f218c2026db8b395e90696d8038f69d9a8", - "size": "430472412" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-b66b5448-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-b66b5448-v1.zip", + "checksum": "SHA-256:a871d945c6bfb685ecff5e30ad759f280c841ea143071466b2e611bd1800f18f", + "size": "430471837" }, { "host": "x86_64-mingw32", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-25c7c119-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-25c7c119-v1.zip", - "checksum": "SHA-256:5760b8d2215c07b0f054e9cd62a268f218c2026db8b395e90696d8038f69d9a8", - "size": "430472412" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-b66b5448-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-b66b5448-v1.zip", + "checksum": "SHA-256:a871d945c6bfb685ecff5e30ad759f280c841ea143071466b2e611bd1800f18f", + "size": "430471837" }, { "host": "arm64-apple-darwin", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-25c7c119-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-25c7c119-v1.zip", - "checksum": "SHA-256:5760b8d2215c07b0f054e9cd62a268f218c2026db8b395e90696d8038f69d9a8", - "size": "430472412" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-b66b5448-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-b66b5448-v1.zip", + "checksum": "SHA-256:a871d945c6bfb685ecff5e30ad759f280c841ea143071466b2e611bd1800f18f", + "size": "430471837" }, { "host": "x86_64-apple-darwin", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-25c7c119-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-25c7c119-v1.zip", - "checksum": "SHA-256:5760b8d2215c07b0f054e9cd62a268f218c2026db8b395e90696d8038f69d9a8", - "size": "430472412" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-b66b5448-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-b66b5448-v1.zip", + "checksum": "SHA-256:a871d945c6bfb685ecff5e30ad759f280c841ea143071466b2e611bd1800f18f", + "size": "430471837" }, { "host": "x86_64-pc-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-25c7c119-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-25c7c119-v1.zip", - "checksum": "SHA-256:5760b8d2215c07b0f054e9cd62a268f218c2026db8b395e90696d8038f69d9a8", - "size": "430472412" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-b66b5448-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-b66b5448-v1.zip", + "checksum": "SHA-256:a871d945c6bfb685ecff5e30ad759f280c841ea143071466b2e611bd1800f18f", + "size": "430471837" }, { "host": "i686-pc-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-25c7c119-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-25c7c119-v1.zip", - "checksum": "SHA-256:5760b8d2215c07b0f054e9cd62a268f218c2026db8b395e90696d8038f69d9a8", - "size": "430472412" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-b66b5448-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-b66b5448-v1.zip", + "checksum": "SHA-256:a871d945c6bfb685ecff5e30ad759f280c841ea143071466b2e611bd1800f18f", + "size": "430471837" }, { "host": "aarch64-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-25c7c119-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-25c7c119-v1.zip", - "checksum": "SHA-256:5760b8d2215c07b0f054e9cd62a268f218c2026db8b395e90696d8038f69d9a8", - "size": "430472412" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-b66b5448-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-b66b5448-v1.zip", + "checksum": "SHA-256:a871d945c6bfb685ecff5e30ad759f280c841ea143071466b2e611bd1800f18f", + "size": "430471837" }, { "host": "arm-linux-gnueabihf", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-25c7c119-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-25c7c119-v1.zip", - "checksum": "SHA-256:5760b8d2215c07b0f054e9cd62a268f218c2026db8b395e90696d8038f69d9a8", - "size": "430472412" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-b66b5448-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-b66b5448-v1.zip", + "checksum": "SHA-256:a871d945c6bfb685ecff5e30ad759f280c841ea143071466b2e611bd1800f18f", + "size": "430471837" } ] }, From a14ce89715088edd5c97d51f4c14f6cd380dded8 Mon Sep 17 00:00:00 2001 From: Lucas Saavedra Vaz <32426024+lucasssvaz@users.noreply.github.com> Date: Tue, 22 Jul 2025 19:38:01 -0300 Subject: [PATCH 138/173] fix(tamc_termod_s3): Fix header includes (#11625) --- variants/tamc_termod_s3/pins_arduino.h | 1 + variants/tamc_termod_s3/variant.cpp | 1 + 2 files changed, 2 insertions(+) diff --git a/variants/tamc_termod_s3/pins_arduino.h b/variants/tamc_termod_s3/pins_arduino.h index 3d846c9b111..89d0b5107ae 100644 --- a/variants/tamc_termod_s3/pins_arduino.h +++ b/variants/tamc_termod_s3/pins_arduino.h @@ -2,6 +2,7 @@ #define Pins_Arduino_h #include +#include #define USB_VID 0x303a #define USB_PID 0x1001 diff --git a/variants/tamc_termod_s3/variant.cpp b/variants/tamc_termod_s3/variant.cpp index 8079bdbba8d..cc255c31684 100644 --- a/variants/tamc_termod_s3/variant.cpp +++ b/variants/tamc_termod_s3/variant.cpp @@ -1,4 +1,5 @@ #include "Arduino.h" +#include "pins_arduino.h" float getBatteryVoltage() { int analogVolt = analogReadMilliVolts(1); From f08efa1fa3dd7c816ddfecc7ef26f771e4aff1a5 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jan=20Proch=C3=A1zka?= <90197375+P-R-O-C-H-Y@users.noreply.github.com> Date: Wed, 23 Jul 2025 00:43:23 +0200 Subject: [PATCH 139/173] feat(docs): Updaze Zigbee docs with latest changes (#11627) --- docs/en/zigbee/ep_fan_control.rst | 133 ++++++++++++++++++++++++++++++ docs/en/zigbee/zigbee_core.rst | 13 +++ docs/en/zigbee/zigbee_ep.rst | 13 +++ 3 files changed, 159 insertions(+) create mode 100644 docs/en/zigbee/ep_fan_control.rst diff --git a/docs/en/zigbee/ep_fan_control.rst b/docs/en/zigbee/ep_fan_control.rst new file mode 100644 index 00000000000..1cfff0820df --- /dev/null +++ b/docs/en/zigbee/ep_fan_control.rst @@ -0,0 +1,133 @@ +################ +ZigbeeFanControl +################ + +About +----- + +The ``ZigbeeFanControl`` class provides a Zigbee endpoint for controlling fan devices in a Home Automation (HA) network. This endpoint implements the Fan Control cluster and allows for controlling fan modes and sequences. + +**Features:** +* Fan mode control (OFF, LOW, MEDIUM, HIGH, ON, AUTO, SMART) +* Fan mode sequence configuration +* State change callbacks +* Zigbee HA standard compliance + +**Use Cases:** +* Smart ceiling fans +* HVAC system fans +* Exhaust fans +* Any device requiring fan speed control + +API Reference +------------- + +Constructor +*********** + +ZigbeeFanControl +^^^^^^^^^^^^^^^^ + +Creates a new Zigbee fan control endpoint. + +.. code-block:: arduino + + ZigbeeFanControl(uint8_t endpoint); + +* ``endpoint`` - Endpoint number (1-254) + +Fan Mode Enums +************** + +ZigbeeFanMode +^^^^^^^^^^^^^ + +Available fan modes for controlling the fan speed and operation. + +.. code-block:: arduino + + enum ZigbeeFanMode { + FAN_MODE_OFF, // Fan is off + FAN_MODE_LOW, // Low speed + FAN_MODE_MEDIUM, // Medium speed + FAN_MODE_HIGH, // High speed + FAN_MODE_ON, // Fan is on (full speed) + FAN_MODE_AUTO, // Automatic mode + FAN_MODE_SMART, // Smart mode + }; + +ZigbeeFanModeSequence +^^^^^^^^^^^^^^^^^^^^^ + +Available fan mode sequences that define which modes are available for the fan. + +.. code-block:: arduino + + enum ZigbeeFanModeSequence { + FAN_MODE_SEQUENCE_LOW_MED_HIGH, // Low -> Medium -> High + FAN_MODE_SEQUENCE_LOW_HIGH, // Low -> High + FAN_MODE_SEQUENCE_LOW_MED_HIGH_AUTO, // Low -> Medium -> High -> Auto + FAN_MODE_SEQUENCE_LOW_HIGH_AUTO, // Low -> High -> Auto + FAN_MODE_SEQUENCE_ON_AUTO, // On -> Auto + }; + +API Methods +*********** + +setFanModeSequence +^^^^^^^^^^^^^^^^^^ + +Sets the fan mode sequence and initializes the fan control. + +.. code-block:: arduino + + bool setFanModeSequence(ZigbeeFanModeSequence sequence); + +* ``sequence`` - The fan mode sequence to set + +This function will return ``true`` if successful, ``false`` otherwise. + +getFanMode +^^^^^^^^^^ + +Gets the current fan mode. + +.. code-block:: arduino + + ZigbeeFanMode getFanMode(); + +This function will return current fan mode. + +getFanModeSequence +^^^^^^^^^^^^^^^^^^ + +Gets the current fan mode sequence. + +.. code-block:: arduino + + ZigbeeFanModeSequence getFanModeSequence(); + +This function will return current fan mode sequence. + +Event Handling +************** + +onFanModeChange +^^^^^^^^^^^^^^^ + +Sets a callback function to be called when the fan mode changes. + +.. code-block:: arduino + + void onFanModeChange(void (*callback)(ZigbeeFanMode mode)); + +* ``callback`` - Function to call when fan mode changes + +Example +------- + +Fan Control Implementation +************************** + +.. literalinclude:: ../../../libraries/Zigbee/examples/Zigbee_Fan_Control/Zigbee_Fan_Control.ino + :language: arduino diff --git a/docs/en/zigbee/zigbee_core.rst b/docs/en/zigbee/zigbee_core.rst index ee8f7234210..89cf88ecca1 100644 --- a/docs/en/zigbee/zigbee_core.rst +++ b/docs/en/zigbee/zigbee_core.rst @@ -345,6 +345,19 @@ Performs a factory reset, clearing all network settings. * ``restart`` - ``true`` to restart after reset (default: ``true``) +onGlobalDefaultResponse +^^^^^^^^^^^^^^^^^^^^^^^ + +Sets a global callback for default response messages. + +.. code-block:: arduino + + void onGlobalDefaultResponse(void (*callback)(zb_cmd_type_t resp_to_cmd, esp_zb_zcl_status_t status, uint8_t endpoint, uint16_t cluster)); + +* ``callback`` - Function pointer to the callback function + +This callback will be called for all endpoints when a default response is received. + Utility Functions ***************** diff --git a/docs/en/zigbee/zigbee_ep.rst b/docs/en/zigbee/zigbee_ep.rst index ce13d60bbaf..10c0c9fd084 100644 --- a/docs/en/zigbee/zigbee_ep.rst +++ b/docs/en/zigbee/zigbee_ep.rst @@ -345,6 +345,19 @@ Sets a callback function for identify events. * ``callback`` - Function to call when identify event occurs * ``time`` - Identify time in seconds +onDefaultResponse +^^^^^^^^^^^^^^^^^ + +Sets a callback for default response messages for this endpoint. + +.. code-block:: arduino + + void onDefaultResponse(void (*callback)(zb_cmd_type_t resp_to_cmd, esp_zb_zcl_status_t status)); + +* ``callback`` - Function pointer to the callback function + +This callback will be called when a default response is received for this specific endpoint. + Supported Endpoints ------------------- From ae634a92e3b3af3a68daf8e79b025a45ca9134ef Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jan=20Proch=C3=A1zka?= <90197375+P-R-O-C-H-Y@users.noreply.github.com> Date: Wed, 23 Jul 2025 00:44:36 +0200 Subject: [PATCH 140/173] fix(zigbee): Fix RGB color calculation (#11624) --- cores/esp32/ColorFormat.c | 12 +++++++----- cores/esp32/ColorFormat.h | 3 ++- libraries/Zigbee/src/ep/ZigbeeColorDimmableLight.cpp | 5 ++--- 3 files changed, 11 insertions(+), 9 deletions(-) diff --git a/cores/esp32/ColorFormat.c b/cores/esp32/ColorFormat.c index a01123545b3..052249f2127 100644 --- a/cores/esp32/ColorFormat.c +++ b/cores/esp32/ColorFormat.c @@ -119,10 +119,10 @@ espHsvColor_t espRgbColorToHsvColor(espRgbColor_t rgb) { } espRgbColor_t espXYColorToRgbColor(uint8_t Level, espXyColor_t xy) { - return espXYToRgbColor(Level, xy.x, xy.y); + return espXYToRgbColor(Level, xy.x, xy.y, true); } -espRgbColor_t espXYToRgbColor(uint8_t Level, uint16_t current_X, uint16_t current_Y) { +espRgbColor_t espXYToRgbColor(uint8_t Level, uint16_t current_X, uint16_t current_Y, bool addXYZScaling) { // convert xyY color space to RGB // https://www.easyrgb.com/en/math.php @@ -156,9 +156,11 @@ espRgbColor_t espXYToRgbColor(uint8_t Level, uint16_t current_X, uint16_t curren // X, Y and Z input refer to a D65/2° standard illuminant. // sR, sG and sB (standard RGB) output range = 0 ÷ 255 // convert XYZ to RGB - CIE XYZ to sRGB - X = X / 100.0f; - Y = Y / 100.0f; - Z = Z / 100.0f; + if (addXYZScaling) { + X = X / 100.0f; + Y = Y / 100.0f; + Z = Z / 100.0f; + } r = (X * 3.2406f) - (Y * 1.5372f) - (Z * 0.4986f); g = -(X * 0.9689f) + (Y * 1.8758f) + (Z * 0.0415f); diff --git a/cores/esp32/ColorFormat.h b/cores/esp32/ColorFormat.h index 0bb87145d16..288b79b5714 100644 --- a/cores/esp32/ColorFormat.h +++ b/cores/esp32/ColorFormat.h @@ -19,6 +19,7 @@ #pragma once #include +#include #ifdef __cplusplus extern "C" { #endif @@ -49,7 +50,7 @@ typedef struct HsvColor_t espHsvColor_t; typedef struct XyColor_t espXyColor_t; typedef struct CtColor_t espCtColor_t; -espRgbColor_t espXYToRgbColor(uint8_t Level, uint16_t current_X, uint16_t current_Y); +espRgbColor_t espXYToRgbColor(uint8_t Level, uint16_t current_X, uint16_t current_Y, bool addXYZScaling); espRgbColor_t espXYColorToRgb(uint8_t Level, espXyColor_t xy); espXyColor_t espRgbColorToXYColor(espRgbColor_t rgb); espXyColor_t espRgbToXYColor(uint8_t r, uint8_t g, uint8_t b); diff --git a/libraries/Zigbee/src/ep/ZigbeeColorDimmableLight.cpp b/libraries/Zigbee/src/ep/ZigbeeColorDimmableLight.cpp index 2fb07fc2187..3611c232c20 100644 --- a/libraries/Zigbee/src/ep/ZigbeeColorDimmableLight.cpp +++ b/libraries/Zigbee/src/ep/ZigbeeColorDimmableLight.cpp @@ -76,14 +76,13 @@ void ZigbeeColorDimmableLight::zbAttributeSet(const esp_zb_zcl_set_attr_value_me return; } else { log_w("Received message ignored. Attribute ID: %d not supported for Level Control", message->attribute.id); - //TODO: implement more attributes -> includes/zcl/esp_zigbee_zcl_level.h } } else if (message->info.cluster == ESP_ZB_ZCL_CLUSTER_ID_COLOR_CONTROL) { if (message->attribute.id == ESP_ZB_ZCL_ATTR_COLOR_CONTROL_CURRENT_X_ID && message->attribute.data.type == ESP_ZB_ZCL_ATTR_TYPE_U16) { 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 + _current_color = espXYToRgbColor(255, light_color_x, light_color_y, false); lightChanged(); return; @@ -91,7 +90,7 @@ 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 + _current_color = espXYToRgbColor(255, light_color_x, light_color_y, false); 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) { From 6fdfccf21b1532dede0e886dd4ab904581a036f3 Mon Sep 17 00:00:00 2001 From: "pre-commit-ci-lite[bot]" <117423508+pre-commit-ci-lite[bot]@users.noreply.github.com> Date: Tue, 22 Jul 2025 22:46:36 +0000 Subject: [PATCH 141/173] ci(pre-commit): Apply automatic fixes --- cores/esp32/esp32-hal-i2c-slave.c | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/cores/esp32/esp32-hal-i2c-slave.c b/cores/esp32/esp32-hal-i2c-slave.c index 79c994869e2..da3a819387a 100644 --- a/cores/esp32/esp32-hal-i2c-slave.c +++ b/cores/esp32/esp32-hal-i2c-slave.c @@ -338,8 +338,7 @@ esp_err_t i2cSlaveInit(uint8_t num, int sda, int scl, uint16_t slaveID, uint32_t } #endif // !defined(CONFIG_IDF_TARGET_ESP32P4) -#if (ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 5, 0)) \ - || (ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 4, 2) && ESP_IDF_VERSION < ESP_IDF_VERSION_VAL(5, 5, 0)) \ +#if (ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 5, 0)) || (ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 4, 2) && ESP_IDF_VERSION < ESP_IDF_VERSION_VAL(5, 5, 0)) \ || (ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 3, 3) && ESP_IDF_VERSION < ESP_IDF_VERSION_VAL(5, 4, 0)) i2c_ll_set_mode(i2c->dev, I2C_BUS_MODE_SLAVE); i2c_ll_enable_pins_open_drain(i2c->dev, true); From e92e631811002adba98eb1e8e0ce167755e0864f Mon Sep 17 00:00:00 2001 From: Angel Nunez Mencias Date: Wed, 23 Jul 2025 07:11:11 +0200 Subject: [PATCH 142/173] feat: Allow calls to timer functions within ISR --- cores/esp32/esp32-hal-timer.c | 19 +++++++------------ 1 file changed, 7 insertions(+), 12 deletions(-) diff --git a/cores/esp32/esp32-hal-timer.c b/cores/esp32/esp32-hal-timer.c index ec6c507358e..508d510508e 100644 --- a/cores/esp32/esp32-hal-timer.c +++ b/cores/esp32/esp32-hal-timer.c @@ -36,9 +36,8 @@ struct timer_struct_t { bool timer_started; }; -inline uint64_t timerRead(hw_timer_t *timer) { +inline IRAM_ATTR uint64_t timerRead(hw_timer_t *timer) { if (timer == NULL) { - log_e("Timer handle is NULL"); return 0; } uint64_t value; @@ -46,15 +45,14 @@ inline uint64_t timerRead(hw_timer_t *timer) { return value; } -void timerWrite(hw_timer_t *timer, uint64_t val) { +void IRAM_ATTR timerWrite(hw_timer_t *timer, uint64_t val) { if (timer == NULL) { - log_e("Timer handle is NULL"); return; } gptimer_set_raw_count(timer->timer_handle, val); } -void timerAlarm(hw_timer_t *timer, uint64_t alarm_value, bool autoreload, uint64_t reload_count) { +void IRAM_ATTR timerAlarm(hw_timer_t *timer, uint64_t alarm_value, bool autoreload, uint64_t reload_count) { if (timer == NULL) { log_e("Timer handle is NULL"); return; @@ -67,7 +65,7 @@ void timerAlarm(hw_timer_t *timer, uint64_t alarm_value, bool autoreload, uint64 }; err = gptimer_set_alarm_action(timer->timer_handle, &alarm_cfg); if (err != ESP_OK) { - log_e("Timer Alarm Write failed, error num=%d", err); + ; // Ignore } } @@ -80,27 +78,24 @@ uint32_t timerGetFrequency(hw_timer_t *timer) { return frequency; } -void timerStart(hw_timer_t *timer) { +void IRAM_ATTR timerStart(hw_timer_t *timer) { if (timer == NULL) { - log_e("Timer handle is NULL"); return; } gptimer_start(timer->timer_handle); timer->timer_started = true; } -void timerStop(hw_timer_t *timer) { +void IRAM_ATTR timerStop(hw_timer_t *timer) { if (timer == NULL) { - log_e("Timer handle is NULL"); return; } gptimer_stop(timer->timer_handle); timer->timer_started = false; } -void timerRestart(hw_timer_t *timer) { +void IRAM_ATTR timerRestart(hw_timer_t *timer) { if (timer == NULL) { - log_e("Timer handle is NULL"); return; } gptimer_set_raw_count(timer->timer_handle, 0); From 67c59a202196abe0029b868c34e57faf4032f0c5 Mon Sep 17 00:00:00 2001 From: Angel Nunez Mencias Date: Wed, 23 Jul 2025 07:26:03 +0200 Subject: [PATCH 143/173] fix: remove log from IRAM function --- cores/esp32/esp32-hal-timer.c | 1 - 1 file changed, 1 deletion(-) diff --git a/cores/esp32/esp32-hal-timer.c b/cores/esp32/esp32-hal-timer.c index 508d510508e..6875b5b2d8f 100644 --- a/cores/esp32/esp32-hal-timer.c +++ b/cores/esp32/esp32-hal-timer.c @@ -54,7 +54,6 @@ void IRAM_ATTR timerWrite(hw_timer_t *timer, uint64_t val) { void IRAM_ATTR timerAlarm(hw_timer_t *timer, uint64_t alarm_value, bool autoreload, uint64_t reload_count) { if (timer == NULL) { - log_e("Timer handle is NULL"); return; } esp_err_t err = ESP_OK; From 3da3ad2f2d38badab90524c7c647681acbd8274e Mon Sep 17 00:00:00 2001 From: Wulu Date: Wed, 23 Jul 2025 20:28:19 +0800 Subject: [PATCH 144/173] feat(board): Add onboard LED support for Waveshare ESP32-S3 Zero (#11630) Defines the standard LED_BUILTIN and RGB_BUILTIN macros for the Waveshare ESP32-S3 Zero, allowing its onboard WS2812 RGB LED to be controlled via standard Arduino APIs. --- variants/waveshare_esp32_s3_zero/pins_arduino.h | 11 ++++++++++- 1 file changed, 10 insertions(+), 1 deletion(-) diff --git a/variants/waveshare_esp32_s3_zero/pins_arduino.h b/variants/waveshare_esp32_s3_zero/pins_arduino.h index 0d73bee16d0..20ce1c88859 100644 --- a/variants/waveshare_esp32_s3_zero/pins_arduino.h +++ b/variants/waveshare_esp32_s3_zero/pins_arduino.h @@ -2,6 +2,7 @@ #define Pins_Arduino_h #include +#include "soc/soc_caps.h" #define USB_VID 0x303a #define USB_PID 0x822B @@ -9,9 +10,17 @@ #define USB_PRODUCT "ESP32-S3-Zero" #define USB_SERIAL "" // Empty string for MAC address -// Partial voltage measurement method +// Onboard WS2812 RGB LED #define WS_RGB 21 +// BUILTIN_LED can be used in new Arduino API digitalWrite() like in Blink.ino +static const uint8_t LED_BUILTIN = SOC_GPIO_PIN_COUNT + WS_RGB; +#define BUILTIN_LED LED_BUILTIN // backward compatibility +#define LED_BUILTIN LED_BUILTIN // allow testing #ifdef LED_BUILTIN +// RGB_BUILTIN and RGB_BRIGHTNESS can be used in new Arduino API rgbLedWrite() +#define RGB_BUILTIN LED_BUILTIN +#define RGB_BRIGHTNESS 64 + // Mapping based on the ESP32S3 data sheet - alternate for OUTPUT static const uint8_t OUTPUT_IO1 = 1; static const uint8_t OUTPUT_IO2 = 2; From c7520ccef0c343611d7f9eb0afb3576f612de2f2 Mon Sep 17 00:00:00 2001 From: Me No Dev Date: Wed, 23 Jul 2025 17:15:59 +0300 Subject: [PATCH 145/173] fix(report): Update Issue-report.yml for version 3.3.0 --- .github/ISSUE_TEMPLATE/Issue-report.yml | 1 + 1 file changed, 1 insertion(+) diff --git a/.github/ISSUE_TEMPLATE/Issue-report.yml b/.github/ISSUE_TEMPLATE/Issue-report.yml index 6dc1b0de171..a16a6ae3d44 100644 --- a/.github/ISSUE_TEMPLATE/Issue-report.yml +++ b/.github/ISSUE_TEMPLATE/Issue-report.yml @@ -43,6 +43,7 @@ body: - latest stable Release (if not listed below) - latest development Release Candidate (RC-X) - latest master (checkout manually) + - v3.3.0 - v3.2.1 - v3.2.0 - v3.1.3 From 126039663f83b3d6f5868a30f7dc2093812ee7b4 Mon Sep 17 00:00:00 2001 From: Angel Nunez Mencias Date: Thu, 24 Jul 2025 00:09:17 +0200 Subject: [PATCH 146/173] fix: conditional logs when not ISR --- cores/esp32/esp32-hal-timer.c | 40 +++++++++++++++++++++++++++++------ 1 file changed, 33 insertions(+), 7 deletions(-) diff --git a/cores/esp32/esp32-hal-timer.c b/cores/esp32/esp32-hal-timer.c index 6875b5b2d8f..38575fc9799 100644 --- a/cores/esp32/esp32-hal-timer.c +++ b/cores/esp32/esp32-hal-timer.c @@ -22,6 +22,12 @@ #include "esp_clk_tree.h" #endif +#if CONFIG_GPTIMER_CTRL_FUNC_IN_IRAM +#define TIMER_IRAM IRAM_ATTR +#else +#define TIMER_IRAM +#endif + typedef void (*voidFuncPtr)(void); typedef void (*voidFuncPtrArg)(void *); @@ -36,8 +42,11 @@ struct timer_struct_t { bool timer_started; }; -inline IRAM_ATTR uint64_t timerRead(hw_timer_t *timer) { +inline TIMER_IRAM uint64_t timerRead(hw_timer_t *timer) { if (timer == NULL) { + #ifndef CONFIG_GPTIMER_CTRL_FUNC_IN_IRAM + log_e("Timer handle is NULL"); + #endif return 0; } uint64_t value; @@ -45,15 +54,21 @@ inline IRAM_ATTR uint64_t timerRead(hw_timer_t *timer) { return value; } -void IRAM_ATTR timerWrite(hw_timer_t *timer, uint64_t val) { +void TIMER_IRAM timerWrite(hw_timer_t *timer, uint64_t val) { if (timer == NULL) { + #ifndef CONFIG_GPTIMER_CTRL_FUNC_IN_IRAM + log_e("Timer handle is NULL"); + #endif return; } gptimer_set_raw_count(timer->timer_handle, val); } -void IRAM_ATTR timerAlarm(hw_timer_t *timer, uint64_t alarm_value, bool autoreload, uint64_t reload_count) { +void TIMER_IRAM timerAlarm(hw_timer_t *timer, uint64_t alarm_value, bool autoreload, uint64_t reload_count) { if (timer == NULL) { + #ifndef CONFIG_GPTIMER_CTRL_FUNC_IN_IRAM + log_e("Timer handle is NULL"); + #endif return; } esp_err_t err = ESP_OK; @@ -64,7 +79,9 @@ void IRAM_ATTR timerAlarm(hw_timer_t *timer, uint64_t alarm_value, bool autorelo }; err = gptimer_set_alarm_action(timer->timer_handle, &alarm_cfg); if (err != ESP_OK) { - ; // Ignore + #ifndef CONFIG_GPTIMER_CTRL_FUNC_IN_IRAM + log_e("Timer Alarm Write failed, error num=%d", err); + #endif } } @@ -77,24 +94,33 @@ uint32_t timerGetFrequency(hw_timer_t *timer) { return frequency; } -void IRAM_ATTR timerStart(hw_timer_t *timer) { +void TIMER_IRAM timerStart(hw_timer_t *timer) { if (timer == NULL) { + #ifndef CONFIG_GPTIMER_CTRL_FUNC_IN_IRAM + log_e("Timer handle is NULL"); + #endif return; } gptimer_start(timer->timer_handle); timer->timer_started = true; } -void IRAM_ATTR timerStop(hw_timer_t *timer) { +void TIMER_IRAM timerStop(hw_timer_t *timer) { if (timer == NULL) { + #ifndef CONFIG_GPTIMER_CTRL_FUNC_IN_IRAM + log_e("Timer handle is NULL"); + #endif return; } gptimer_stop(timer->timer_handle); timer->timer_started = false; } -void IRAM_ATTR timerRestart(hw_timer_t *timer) { +void TIMER_IRAM timerRestart(hw_timer_t *timer) { if (timer == NULL) { + #ifndef CONFIG_GPTIMER_CTRL_FUNC_IN_IRAM + log_e("Timer handle is NULL"); + #endif return; } gptimer_set_raw_count(timer->timer_handle, 0); From e3a0a3a6c58eca709dacf6f1d2e7cd169f522e86 Mon Sep 17 00:00:00 2001 From: "pre-commit-ci-lite[bot]" <117423508+pre-commit-ci-lite[bot]@users.noreply.github.com> Date: Thu, 24 Jul 2025 11:22:13 +0000 Subject: [PATCH 147/173] ci(pre-commit): Apply automatic fixes --- cores/esp32/esp32-hal-timer.c | 28 ++++++++++++++-------------- 1 file changed, 14 insertions(+), 14 deletions(-) diff --git a/cores/esp32/esp32-hal-timer.c b/cores/esp32/esp32-hal-timer.c index 38575fc9799..85e007143bd 100644 --- a/cores/esp32/esp32-hal-timer.c +++ b/cores/esp32/esp32-hal-timer.c @@ -44,9 +44,9 @@ struct timer_struct_t { inline TIMER_IRAM uint64_t timerRead(hw_timer_t *timer) { if (timer == NULL) { - #ifndef CONFIG_GPTIMER_CTRL_FUNC_IN_IRAM +#ifndef CONFIG_GPTIMER_CTRL_FUNC_IN_IRAM log_e("Timer handle is NULL"); - #endif +#endif return 0; } uint64_t value; @@ -56,9 +56,9 @@ inline TIMER_IRAM uint64_t timerRead(hw_timer_t *timer) { void TIMER_IRAM timerWrite(hw_timer_t *timer, uint64_t val) { if (timer == NULL) { - #ifndef CONFIG_GPTIMER_CTRL_FUNC_IN_IRAM +#ifndef CONFIG_GPTIMER_CTRL_FUNC_IN_IRAM log_e("Timer handle is NULL"); - #endif +#endif return; } gptimer_set_raw_count(timer->timer_handle, val); @@ -66,9 +66,9 @@ void TIMER_IRAM timerWrite(hw_timer_t *timer, uint64_t val) { void TIMER_IRAM timerAlarm(hw_timer_t *timer, uint64_t alarm_value, bool autoreload, uint64_t reload_count) { if (timer == NULL) { - #ifndef CONFIG_GPTIMER_CTRL_FUNC_IN_IRAM +#ifndef CONFIG_GPTIMER_CTRL_FUNC_IN_IRAM log_e("Timer handle is NULL"); - #endif +#endif return; } esp_err_t err = ESP_OK; @@ -79,9 +79,9 @@ void TIMER_IRAM timerAlarm(hw_timer_t *timer, uint64_t alarm_value, bool autorel }; err = gptimer_set_alarm_action(timer->timer_handle, &alarm_cfg); if (err != ESP_OK) { - #ifndef CONFIG_GPTIMER_CTRL_FUNC_IN_IRAM +#ifndef CONFIG_GPTIMER_CTRL_FUNC_IN_IRAM log_e("Timer Alarm Write failed, error num=%d", err); - #endif +#endif } } @@ -96,9 +96,9 @@ uint32_t timerGetFrequency(hw_timer_t *timer) { void TIMER_IRAM timerStart(hw_timer_t *timer) { if (timer == NULL) { - #ifndef CONFIG_GPTIMER_CTRL_FUNC_IN_IRAM +#ifndef CONFIG_GPTIMER_CTRL_FUNC_IN_IRAM log_e("Timer handle is NULL"); - #endif +#endif return; } gptimer_start(timer->timer_handle); @@ -107,9 +107,9 @@ void TIMER_IRAM timerStart(hw_timer_t *timer) { void TIMER_IRAM timerStop(hw_timer_t *timer) { if (timer == NULL) { - #ifndef CONFIG_GPTIMER_CTRL_FUNC_IN_IRAM +#ifndef CONFIG_GPTIMER_CTRL_FUNC_IN_IRAM log_e("Timer handle is NULL"); - #endif +#endif return; } gptimer_stop(timer->timer_handle); @@ -118,9 +118,9 @@ void TIMER_IRAM timerStop(hw_timer_t *timer) { void TIMER_IRAM timerRestart(hw_timer_t *timer) { if (timer == NULL) { - #ifndef CONFIG_GPTIMER_CTRL_FUNC_IN_IRAM +#ifndef CONFIG_GPTIMER_CTRL_FUNC_IN_IRAM log_e("Timer handle is NULL"); - #endif +#endif return; } gptimer_set_raw_count(timer->timer_handle, 0); From 5c24611b913e90d65d75b5208359f6364a2509f9 Mon Sep 17 00:00:00 2001 From: Parsaabasi Date: Fri, 25 Jul 2025 12:35:55 +0200 Subject: [PATCH 148/173] feat(github): remove ESP32 forum from issue template --- .github/ISSUE_TEMPLATE/config.yml | 3 --- 1 file changed, 3 deletions(-) diff --git a/.github/ISSUE_TEMPLATE/config.yml b/.github/ISSUE_TEMPLATE/config.yml index e879b09bec2..060397abd1e 100644 --- a/.github/ISSUE_TEMPLATE/config.yml +++ b/.github/ISSUE_TEMPLATE/config.yml @@ -3,6 +3,3 @@ contact_links: - name: Arduino Core for Espressif Discord Server url: https://discord.gg/8xY6e9crwv about: Community Discord server for questions and help - - name: ESP32 Forum - Arduino - url: https://esp32.com/viewforum.php?f=19 - about: Official Forum for questions From 10e73be7146d6b6d80777016bdc4ddb810cf56a7 Mon Sep 17 00:00:00 2001 From: Parsaabasi Date: Fri, 25 Jul 2025 12:49:46 +0200 Subject: [PATCH 149/173] feat(github): Update template type --- .github/ISSUE_TEMPLATE/Issue-report.yml | 11 +++++++++++ 1 file changed, 11 insertions(+) diff --git a/.github/ISSUE_TEMPLATE/Issue-report.yml b/.github/ISSUE_TEMPLATE/Issue-report.yml index a16a6ae3d44..864ef65b92b 100644 --- a/.github/ISSUE_TEMPLATE/Issue-report.yml +++ b/.github/ISSUE_TEMPLATE/Issue-report.yml @@ -80,6 +80,17 @@ body: - other validations: required: true + - type: dropdown + id: type + attributes: + label: Type + description: How would you define the type of the issue? Please select from the types below. + options: + - Task + - Bug + - Question + validations: + required: true - type: input id: IDE attributes: From b3152a4ffc38f374cb70ff999f3c64a9bef75297 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jan=20Proch=C3=A1zka?= <90197375+P-R-O-C-H-Y@users.noreply.github.com> Date: Fri, 25 Jul 2025 14:10:32 +0200 Subject: [PATCH 150/173] feat(zigbee): Update SDK to latest --- idf_component.yml | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/idf_component.yml b/idf_component.yml index 82f14ea554a..f9357a401f9 100644 --- a/idf_component.yml +++ b/idf_component.yml @@ -54,12 +54,12 @@ dependencies: espressif/esp_modem: version: "^1.1.0" espressif/esp-zboss-lib: - version: "==1.6.4" # compatible with esp-zigbee-lib 1.6.5 + version: "==1.6.4" # compatible with esp-zigbee-lib 1.6.6 require: public rules: - if: "target not in [esp32c2, esp32p4]" espressif/esp-zigbee-lib: - version: "==1.6.5" + version: "==1.6.6" require: public rules: - if: "target not in [esp32c2, esp32p4]" From b139bae5028ac2be25cd91943f62c7546b307ea1 Mon Sep 17 00:00:00 2001 From: Wulu Date: Sun, 27 Jul 2025 19:50:20 +0800 Subject: [PATCH 151/173] docs: Remove incorrect statement on ESP32-S3 default resolution --- docs/en/api/adc.rst | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/docs/en/api/adc.rst b/docs/en/api/adc.rst index 434384f8d9b..9c16db54ec1 100644 --- a/docs/en/api/adc.rst +++ b/docs/en/api/adc.rst @@ -52,8 +52,7 @@ This function will return analog value in millivolts (calibrated). analogReadResolution ^^^^^^^^^^^^^^^^^^^^ -This function is used to set the resolution of ``analogRead`` return value. Default is 12 bits (range from 0 to 4095) -for all chips except ESP32-S3 where default is 13 bits (range from 0 to 8191). +This function is used to set the resolution of ``analogRead`` return value. Default is 12 bits (range from 0 to 4095). When different resolution is set, the values read will be shifted to match the given resolution. Range is 1 - 16 .The default value will be used, if this function is not used. From ed6e95d4fc207575d932dc12a7783eff724280f4 Mon Sep 17 00:00:00 2001 From: Paula Scharf Date: Mon, 28 Jul 2025 08:46:14 +0200 Subject: [PATCH 152/173] feat(board): use existing led API --- variants/sensebox_eye/variant.cpp | 46 +++---------------------------- 1 file changed, 4 insertions(+), 42 deletions(-) diff --git a/variants/sensebox_eye/variant.cpp b/variants/sensebox_eye/variant.cpp index d8a37f58758..23bc1d385f7 100644 --- a/variants/sensebox_eye/variant.cpp +++ b/variants/sensebox_eye/variant.cpp @@ -1,50 +1,12 @@ #include "esp32-hal-gpio.h" #include "pins_arduino.h" -#include "driver/rmt_tx.h" extern "C" { -void blinkLED(uint8_t color[3], rmt_channel_handle_t led_chan, rmt_encoder_handle_t ws2812_encoder, rmt_transmit_config_t tx_config) { - ESP_ERROR_CHECK(rmt_transmit(led_chan, ws2812_encoder, color, sizeof(color), &tx_config)); - rmt_tx_wait_all_done(led_chan, portMAX_DELAY); - - // Wait a moment - delay(50); - - // Turn LED off - uint8_t pixel_off[3] = { 0x00, 0x00, 0x00 }; - ESP_ERROR_CHECK(rmt_transmit(led_chan, ws2812_encoder, pixel_off, sizeof(pixel_off), &tx_config)); - rmt_tx_wait_all_done(led_chan, portMAX_DELAY); -} - void initVariant(void) { - rmt_channel_handle_t led_chan = NULL; - rmt_tx_channel_config_t tx_chan_config = {}; - tx_chan_config.clk_src = RMT_CLK_SRC_DEFAULT; - tx_chan_config.resolution_hz = 10 * 1000 * 1000; - tx_chan_config.mem_block_symbols = 64; - tx_chan_config.trans_queue_depth = 4; - tx_chan_config.gpio_num = (gpio_num_t)PIN_LED; - tx_chan_config.flags.invert_out = false; - tx_chan_config.flags.with_dma = false; - ESP_ERROR_CHECK(rmt_new_tx_channel(&tx_chan_config, &led_chan)); - - // WS2812 encoder config (available in `esp-rmt`) - rmt_encoder_handle_t ws2812_encoder = NULL; - rmt_bytes_encoder_config_t bytes_encoder_config = { - .bit0 = { .duration0 = 4, .level0 = 1, .duration1 = 9, .level1 = 0 }, - .bit1 = { .duration0 = 8, .level0 = 1, .duration1 = 5, .level1 = 0 }, - .flags = { .msb_first = true } - }; - ESP_ERROR_CHECK(rmt_new_bytes_encoder(&bytes_encoder_config, &ws2812_encoder)); - - ESP_ERROR_CHECK(rmt_enable(led_chan)); - - rmt_transmit_config_t tx_config = { - .loop_count = 0 - }; - - uint8_t pixel[3] = { 0x10, 0x00, 0x00 }; // green - blinkLED(pixel, led_chan, ws2812_encoder, tx_config); + // blink the RGB LED + rgbLedWrite(PIN_LED, 0x00, 0x10, 0x00); // green + delay(20); + rgbLedWrite(PIN_LED, 0x00, 0x00, 0x00); // off } } From 0dcb1d1178c0d8c73870a83f892fada1114d8fc0 Mon Sep 17 00:00:00 2001 From: "pre-commit-ci-lite[bot]" <117423508+pre-commit-ci-lite[bot]@users.noreply.github.com> Date: Mon, 28 Jul 2025 06:56:12 +0000 Subject: [PATCH 153/173] ci(pre-commit): Apply automatic fixes --- .../sensebox_eye/partitions-16MB-tinyuf2.csv | 2 +- variants/sensebox_eye/pins_arduino.h | 26 +++++++++---------- variants/sensebox_eye/variant.cpp | 4 +-- variants/sensebox_mcu_esp32s2/APOTA.ino | 4 +-- 4 files changed, 18 insertions(+), 18 deletions(-) diff --git a/variants/sensebox_eye/partitions-16MB-tinyuf2.csv b/variants/sensebox_eye/partitions-16MB-tinyuf2.csv index 960f3fbbd71..55dc6213161 100644 --- a/variants/sensebox_eye/partitions-16MB-tinyuf2.csv +++ b/variants/sensebox_eye/partitions-16MB-tinyuf2.csv @@ -7,4 +7,4 @@ otadata, data, ota, 0xe000, 8K, ota_0, 0, ota_0, 0x10000, 2048K, ota_1, 0, ota_1, 0x210000, 2048K, uf2, app, factory,0x410000, 256K, -ffat, data, fat, 0x450000, 11968K, \ No newline at end of file +ffat, data, fat, 0x450000, 11968K, diff --git a/variants/sensebox_eye/pins_arduino.h b/variants/sensebox_eye/pins_arduino.h index 99984dbc6a4..70c2bd880c9 100644 --- a/variants/sensebox_eye/pins_arduino.h +++ b/variants/sensebox_eye/pins_arduino.h @@ -10,15 +10,15 @@ #define USB_SERIAL "" // Empty string for MAC address // Default USB FirmwareMSC Settings -#define USB_FW_MSC_VENDOR_ID "senseBox" // max 8 chars +#define USB_FW_MSC_VENDOR_ID "senseBox" // max 8 chars #define USB_FW_MSC_PRODUCT_ID "Eye ESP32S3" // max 16 chars -#define USB_FW_MSC_PRODUCT_REVISION "1.00" // max 4 chars -#define USB_FW_MSC_VOLUME_NAME "senseBox" // max 11 chars +#define USB_FW_MSC_PRODUCT_REVISION "1.00" // max 4 chars +#define USB_FW_MSC_VOLUME_NAME "senseBox" // max 11 chars #define USB_FW_MSC_SERIAL_NUMBER 0x00000000 #define PIN_RGB_LED 45 // RGB LED #define RGBLED_PIN 45 // RGB LED -#define PIN_LED 45 +#define PIN_LED 45 #define RGBLED_NUM 1 // number of RGB LEDs // Default I2C QWIIC-Ports @@ -28,14 +28,14 @@ static const uint8_t SCL = 1; #define PIN_QWIIC_SCL 1 // IO Pins -#define PIN_IO14 14 -static const uint8_t A14 = PIN_IO14; // Analog -static const uint8_t D14 = PIN_IO14; // Digital -static const uint8_t T14 = PIN_IO14; // Touch -#define PIN_IO48 48 -static const uint8_t A48 = PIN_IO48; // Analog -static const uint8_t D48 = PIN_IO48; // Digital -static const uint8_t T48 = PIN_IO48; // Touch +#define PIN_IO14 14 +static const uint8_t A14 = PIN_IO14; // Analog +static const uint8_t D14 = PIN_IO14; // Digital +static const uint8_t T14 = PIN_IO14; // Touch +#define PIN_IO48 48 +static const uint8_t A48 = PIN_IO48; // Analog +static const uint8_t D48 = PIN_IO48; // Digital +static const uint8_t T48 = PIN_IO48; // Touch // Button #define PIN_BUTTON 47 @@ -87,4 +87,4 @@ static const uint8_t RX = 44; #define LORA_TX 43 #define LORA_RX 44 -#endif /* Pins_Arduino_h */ \ No newline at end of file +#endif /* Pins_Arduino_h */ diff --git a/variants/sensebox_eye/variant.cpp b/variants/sensebox_eye/variant.cpp index 23bc1d385f7..4d6e38e73e0 100644 --- a/variants/sensebox_eye/variant.cpp +++ b/variants/sensebox_eye/variant.cpp @@ -5,8 +5,8 @@ extern "C" { void initVariant(void) { // blink the RGB LED - rgbLedWrite(PIN_LED, 0x00, 0x10, 0x00); // green + rgbLedWrite(PIN_LED, 0x00, 0x10, 0x00); // green delay(20); - rgbLedWrite(PIN_LED, 0x00, 0x00, 0x00); // off + rgbLedWrite(PIN_LED, 0x00, 0x00, 0x00); // off } } diff --git a/variants/sensebox_mcu_esp32s2/APOTA.ino b/variants/sensebox_mcu_esp32s2/APOTA.ino index afb741499b2..114f0325e92 100644 --- a/variants/sensebox_mcu_esp32s2/APOTA.ino +++ b/variants/sensebox_mcu_esp32s2/APOTA.ino @@ -1,5 +1,5 @@ -// APOTA is a arduino fallback sketch that is written to OTA1_Partition. -// APOTA opens an access point which waits to receive a .binfile on /sketch. +// APOTA is a arduino fallback sketch that is written to OTA1_Partition. +// APOTA opens an access point which waits to receive a .binfile on /sketch. // After succesful upload, the file is written to OTA0_Partition and the microcontroller reboots to the newly uploaded sketch. #define DISPLAY_ENABLED From e375698adeb0d3ef306782b1a669bf7a60d6b190 Mon Sep 17 00:00:00 2001 From: Paula Scharf Date: Mon, 28 Jul 2025 09:44:04 +0200 Subject: [PATCH 154/173] feat(board): spelling --- variants/sensebox_mcu_esp32s2/APOTA.ino | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/variants/sensebox_mcu_esp32s2/APOTA.ino b/variants/sensebox_mcu_esp32s2/APOTA.ino index 114f0325e92..a1fbab402ca 100644 --- a/variants/sensebox_mcu_esp32s2/APOTA.ino +++ b/variants/sensebox_mcu_esp32s2/APOTA.ino @@ -1,6 +1,6 @@ -// APOTA is a arduino fallback sketch that is written to OTA1_Partition. -// APOTA opens an access point which waits to receive a .binfile on /sketch. -// After succesful upload, the file is written to OTA0_Partition and the microcontroller reboots to the newly uploaded sketch. +// APOTA is an Arduino fallback sketch that is written to OTA1_Partition. +// APOTA opens an access point which waits to receive a .bin file on /sketch. +// After successful upload, the file is written to OTA0_Partition, and the microcontroller reboots to the newly uploaded sketch. #define DISPLAY_ENABLED From a5e85f675e09aa9e5532dd9e4e24019e243f3ee8 Mon Sep 17 00:00:00 2001 From: "pre-commit-ci-lite[bot]" <117423508+pre-commit-ci-lite[bot]@users.noreply.github.com> Date: Mon, 28 Jul 2025 07:44:47 +0000 Subject: [PATCH 155/173] ci(pre-commit): Apply automatic fixes --- variants/sensebox_mcu_esp32s2/APOTA.ino | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/variants/sensebox_mcu_esp32s2/APOTA.ino b/variants/sensebox_mcu_esp32s2/APOTA.ino index a1fbab402ca..67348d0b20f 100644 --- a/variants/sensebox_mcu_esp32s2/APOTA.ino +++ b/variants/sensebox_mcu_esp32s2/APOTA.ino @@ -1,5 +1,5 @@ -// APOTA is an Arduino fallback sketch that is written to OTA1_Partition. -// APOTA opens an access point which waits to receive a .bin file on /sketch. +// APOTA is an Arduino fallback sketch that is written to OTA1_Partition. +// APOTA opens an access point which waits to receive a .bin file on /sketch. // After successful upload, the file is written to OTA0_Partition, and the microcontroller reboots to the newly uploaded sketch. #define DISPLAY_ENABLED From 311955ccbb0fa657ff6e03fcdd553bd5094c826a Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jan=20Proch=C3=A1zka?= <90197375+P-R-O-C-H-Y@users.noreply.github.com> Date: Mon, 28 Jul 2025 11:59:01 +0200 Subject: [PATCH 156/173] fix(component): Use external diag logs to avoid error --- idf_component_examples/esp_matter_light/sdkconfig.defaults | 5 +++++ idf_component_examples/hello_world/sdkconfig.defaults | 5 +++++ idf_component_examples/hw_cdc_hello_world/sdkconfig.defaults | 5 +++++ 3 files changed, 15 insertions(+) diff --git a/idf_component_examples/esp_matter_light/sdkconfig.defaults b/idf_component_examples/esp_matter_light/sdkconfig.defaults index 43871661856..8688318fa36 100644 --- a/idf_component_examples/esp_matter_light/sdkconfig.defaults +++ b/idf_component_examples/esp_matter_light/sdkconfig.defaults @@ -60,3 +60,8 @@ CONFIG_MBEDTLS_HKDF_C=y # Increase LwIP IPv6 address number to 6 (MAX_FABRIC + 1) # unique local addresses for fabrics(MAX_FABRIC), a link local address(1) CONFIG_LWIP_IPV6_NUM_ADDRESSES=6 + +# +# DIAGNOSTICS +# +CONFIG_DIAG_USE_EXTERNAL_LOG_WRAP=y diff --git a/idf_component_examples/hello_world/sdkconfig.defaults b/idf_component_examples/hello_world/sdkconfig.defaults index bb723653f8a..a604d9767fd 100644 --- a/idf_component_examples/hello_world/sdkconfig.defaults +++ b/idf_component_examples/hello_world/sdkconfig.defaults @@ -10,3 +10,8 @@ CONFIG_AUTOSTART_ARDUINO=y CONFIG_FREERTOS_HZ=1000 # end of FREERTOS # end of Component config + +# +# DIAGNOSTICS +# +CONFIG_DIAG_USE_EXTERNAL_LOG_WRAP=y diff --git a/idf_component_examples/hw_cdc_hello_world/sdkconfig.defaults b/idf_component_examples/hw_cdc_hello_world/sdkconfig.defaults index bb723653f8a..a604d9767fd 100644 --- a/idf_component_examples/hw_cdc_hello_world/sdkconfig.defaults +++ b/idf_component_examples/hw_cdc_hello_world/sdkconfig.defaults @@ -10,3 +10,8 @@ CONFIG_AUTOSTART_ARDUINO=y CONFIG_FREERTOS_HZ=1000 # end of FREERTOS # end of Component config + +# +# DIAGNOSTICS +# +CONFIG_DIAG_USE_EXTERNAL_LOG_WRAP=y From 6cc7a63cd021e6cd457d086dc1efb6b550554c65 Mon Sep 17 00:00:00 2001 From: Wulu Date: Mon, 28 Jul 2025 22:43:55 +0800 Subject: [PATCH 157/173] docs: Change ADC default resolution exception to ESP32-S2 --- docs/en/api/adc.rst | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/docs/en/api/adc.rst b/docs/en/api/adc.rst index 9c16db54ec1..416a8afd803 100644 --- a/docs/en/api/adc.rst +++ b/docs/en/api/adc.rst @@ -52,7 +52,8 @@ This function will return analog value in millivolts (calibrated). analogReadResolution ^^^^^^^^^^^^^^^^^^^^ -This function is used to set the resolution of ``analogRead`` return value. Default is 12 bits (range from 0 to 4095). +This function is used to set the resolution of ``analogRead`` return value. Default is 12 bits (range from 0 to 4095) +for all chips except ESP32-S2 where default is 13 bits (range from 0 to 8191). When different resolution is set, the values read will be shifted to match the given resolution. Range is 1 - 16 .The default value will be used, if this function is not used. From f5b08cd81297136fbf0121835858bca39bd4d394 Mon Sep 17 00:00:00 2001 From: Lucas Saavedra Vaz <32426024+lucasssvaz@users.noreply.github.com> Date: Mon, 28 Jul 2025 13:38:14 -0300 Subject: [PATCH 158/173] feat(codeql): Add CodeQL analysis for interpreted languages (#11662) --- .github/CODEOWNERS | 1 + .github/codeql/codeql-config.yml | 26 +++++++++++++++++++++++ .github/workflows/codeql_actions.yml | 31 ++++++++++++++++++++++++++++ .github/workflows/codeql_python.yml | 30 +++++++++++++++++++++++++++ 4 files changed, 88 insertions(+) create mode 100644 .github/codeql/codeql-config.yml create mode 100644 .github/workflows/codeql_actions.yml create mode 100644 .github/workflows/codeql_python.yml diff --git a/.github/CODEOWNERS b/.github/CODEOWNERS index 64d241ba20a..d07ef88044d 100644 --- a/.github/CODEOWNERS +++ b/.github/CODEOWNERS @@ -11,6 +11,7 @@ # CI /.github/ @lucasssvaz @me-no-dev @P-R-O-C-H-Y +/.github/codeql/ @lucasssvaz /.gitlab/ @lucasssvaz /tests/ @lucasssvaz @P-R-O-C-H-Y diff --git a/.github/codeql/codeql-config.yml b/.github/codeql/codeql-config.yml new file mode 100644 index 00000000000..c640d7bd6ef --- /dev/null +++ b/.github/codeql/codeql-config.yml @@ -0,0 +1,26 @@ +name: "CodeQL config" + +packs: + - trailofbits/cpp-queries + - githubsecuritylab/codeql-cpp-queries + - githubsecuritylab/codeql-python-queries + +queries: + - uses: security-extended + - uses: security-and-quality + +query-filters: + - exclude: + query path: + - /^experimental\/.*/ + - exclude: + tags contain: + - experimental + - exclude: + problem.severity: + - recommendation + - exclude: + id: tob/cpp/use-of-legacy-algorithm + +paths-ignore: + - tests/** diff --git a/.github/workflows/codeql_actions.yml b/.github/workflows/codeql_actions.yml new file mode 100644 index 00000000000..07688f896ba --- /dev/null +++ b/.github/workflows/codeql_actions.yml @@ -0,0 +1,31 @@ +name: CodeQL Actions Analysis + +on: + workflow_dispatch: + push: + branches: + - master + pull_request: + paths: + - ".github/workflows/*.yml" + - ".github/workflows/*.yaml" + +jobs: + codeql-analysis: + name: CodeQL Actions Analysis + runs-on: ubuntu-latest + + steps: + - name: Checkout repository + uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683 # v4.2.2 + + - name: Initialize CodeQL + uses: github/codeql-action/init@181d5eefc20863364f96762470ba6f862bdef56b # v3.29.2 + with: + languages: actions + config-file: ./.github/codeql/codeql-config.yml + + - name: Run CodeQL Analysis + uses: github/codeql-action/analyze@181d5eefc20863364f96762470ba6f862bdef56b # v3.29.2 + with: + category: "Analysis: Actions" diff --git a/.github/workflows/codeql_python.yml b/.github/workflows/codeql_python.yml new file mode 100644 index 00000000000..312974cd3c1 --- /dev/null +++ b/.github/workflows/codeql_python.yml @@ -0,0 +1,30 @@ +name: CodeQL Python Analysis + +on: + workflow_dispatch: + push: + branches: + - master + pull_request: + paths: + - "**/*.py" + +jobs: + codeql-analysis: + name: CodeQL Python Analysis + runs-on: ubuntu-latest + + steps: + - name: Checkout repository + uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683 # v4.2.2 + + - name: Initialize CodeQL + uses: github/codeql-action/init@181d5eefc20863364f96762470ba6f862bdef56b # v3.29.2 + with: + languages: python + config-file: ./.github/codeql/codeql-config.yml + + - name: Run CodeQL Analysis + uses: github/codeql-action/analyze@181d5eefc20863364f96762470ba6f862bdef56b # v3.29.2 + with: + category: "Analysis: Python" From 18daf31211eaba8ea7108263a938c6672e4c036a Mon Sep 17 00:00:00 2001 From: Parsaabasi Date: Tue, 29 Jul 2025 14:22:58 +0200 Subject: [PATCH 159/173] fix(github-template): indentation --- .github/ISSUE_TEMPLATE/Issue-report.yml | 16 ++++++++-------- 1 file changed, 8 insertions(+), 8 deletions(-) diff --git a/.github/ISSUE_TEMPLATE/Issue-report.yml b/.github/ISSUE_TEMPLATE/Issue-report.yml index 864ef65b92b..e1c13413764 100644 --- a/.github/ISSUE_TEMPLATE/Issue-report.yml +++ b/.github/ISSUE_TEMPLATE/Issue-report.yml @@ -81,14 +81,14 @@ body: validations: required: true - type: dropdown - id: type - attributes: - label: Type - description: How would you define the type of the issue? Please select from the types below. - options: - - Task - - Bug - - Question + id: type + attributes: + label: Type + description: How would you define the type of the issue? Please select from the types below. + options: + - "Task" + - "Bug" + - "Question" validations: required: true - type: input From 866359a0e38f726b539093237d6d688003f8ba96 Mon Sep 17 00:00:00 2001 From: Lucas Saavedra Vaz <32426024+lucasssvaz@users.noreply.github.com> Date: Tue, 29 Jul 2025 11:54:40 -0300 Subject: [PATCH 160/173] fix(codeql): Merge CodeQL workflows to avoid errors in PRs --- .../{codeql_actions.yml => codeql.yml} | 12 +++++--- .github/workflows/codeql_python.yml | 30 ------------------- 2 files changed, 8 insertions(+), 34 deletions(-) rename .github/workflows/{codeql_actions.yml => codeql.yml} (72%) delete mode 100644 .github/workflows/codeql_python.yml diff --git a/.github/workflows/codeql_actions.yml b/.github/workflows/codeql.yml similarity index 72% rename from .github/workflows/codeql_actions.yml rename to .github/workflows/codeql.yml index 07688f896ba..fb0badd78ec 100644 --- a/.github/workflows/codeql_actions.yml +++ b/.github/workflows/codeql.yml @@ -1,4 +1,4 @@ -name: CodeQL Actions Analysis +name: CodeQL Analysis on: workflow_dispatch: @@ -7,13 +7,17 @@ on: - master pull_request: paths: + - "**/*.py" - ".github/workflows/*.yml" - ".github/workflows/*.yaml" jobs: codeql-analysis: - name: CodeQL Actions Analysis + name: CodeQL ${{ matrix.language }} analysis runs-on: ubuntu-latest + strategy: + matrix: + language: [python, actions] steps: - name: Checkout repository @@ -22,10 +26,10 @@ jobs: - name: Initialize CodeQL uses: github/codeql-action/init@181d5eefc20863364f96762470ba6f862bdef56b # v3.29.2 with: - languages: actions + languages: ${{ matrix.language }} config-file: ./.github/codeql/codeql-config.yml - name: Run CodeQL Analysis uses: github/codeql-action/analyze@181d5eefc20863364f96762470ba6f862bdef56b # v3.29.2 with: - category: "Analysis: Actions" + category: "/language:${{ matrix.language }}" diff --git a/.github/workflows/codeql_python.yml b/.github/workflows/codeql_python.yml deleted file mode 100644 index 312974cd3c1..00000000000 --- a/.github/workflows/codeql_python.yml +++ /dev/null @@ -1,30 +0,0 @@ -name: CodeQL Python Analysis - -on: - workflow_dispatch: - push: - branches: - - master - pull_request: - paths: - - "**/*.py" - -jobs: - codeql-analysis: - name: CodeQL Python Analysis - runs-on: ubuntu-latest - - steps: - - name: Checkout repository - uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683 # v4.2.2 - - - name: Initialize CodeQL - uses: github/codeql-action/init@181d5eefc20863364f96762470ba6f862bdef56b # v3.29.2 - with: - languages: python - config-file: ./.github/codeql/codeql-config.yml - - - name: Run CodeQL Analysis - uses: github/codeql-action/analyze@181d5eefc20863364f96762470ba6f862bdef56b # v3.29.2 - with: - category: "Analysis: Python" From 4eff7f9ae408a9fdd5d15c86184bda785d881aa3 Mon Sep 17 00:00:00 2001 From: Lucas Saavedra Vaz <32426024+lucasssvaz@users.noreply.github.com> Date: Tue, 29 Jul 2025 14:43:25 -0300 Subject: [PATCH 161/173] fix(template): Fix issue report template --- .github/ISSUE_TEMPLATE/Issue-report.yml | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/.github/ISSUE_TEMPLATE/Issue-report.yml b/.github/ISSUE_TEMPLATE/Issue-report.yml index e1c13413764..97834925020 100644 --- a/.github/ISSUE_TEMPLATE/Issue-report.yml +++ b/.github/ISSUE_TEMPLATE/Issue-report.yml @@ -80,12 +80,12 @@ body: - other validations: required: true - - type: dropdown - id: type - attributes: - label: Type - description: How would you define the type of the issue? Please select from the types below. - options: + - type: dropdown + id: type + attributes: + label: Type + description: How would you define the type of the issue? Please select from the types below. + options: - "Task" - "Bug" - "Question" From 2b9d7576fa918e3f27ebd743690489c0874c2309 Mon Sep 17 00:00:00 2001 From: Lucas Saavedra Vaz <32426024+lucasssvaz@users.noreply.github.com> Date: Wed, 30 Jul 2025 10:27:31 -0300 Subject: [PATCH 162/173] ci(component): Move component compilation to a separate workflow --- .github/scripts/set_push_chunks.sh | 1 - .github/workflows/build_component.yml | 133 ++++++++++++++++++++++++++ .github/workflows/push.yml | 120 +++++------------------ .github/workflows/tests.yml | 11 +-- 4 files changed, 159 insertions(+), 106 deletions(-) create mode 100644 .github/workflows/build_component.yml diff --git a/.github/scripts/set_push_chunks.sh b/.github/scripts/set_push_chunks.sh index ff0af7da6e8..69cd9a7f7de 100644 --- a/.github/scripts/set_push_chunks.sh +++ b/.github/scripts/set_push_chunks.sh @@ -78,7 +78,6 @@ chunks+="]" echo "build_all=$build_all" echo "build_libraries=$BUILD_LIBRARIES" echo "build_static_sketches=$BUILD_STATIC_SKETCHES" - echo "build_idf=$BUILD_IDF" echo "chunk_count=$chunks_count" echo "chunks=$chunks" } >> "$GITHUB_OUTPUT" diff --git a/.github/workflows/build_component.yml b/.github/workflows/build_component.yml new file mode 100644 index 00000000000..5553b4b2024 --- /dev/null +++ b/.github/workflows/build_component.yml @@ -0,0 +1,133 @@ +name: Arduino as ESP-IDF Component + +on: + workflow_dispatch: + inputs: + idf_ver: + description: "IDF Versions" + default: "release-v5.3,release-v5.4,release-v5.5" + type: "string" + required: true + idf_targets: + description: "IDF Targets" + default: "esp32,esp32s2,esp32s3,esp32c2,esp32c3,esp32c6,esp32h2,esp32p4" + type: "string" + required: true + push: + branches: + - master + - release/* + pull_request: + paths: + - "cores/**" + - "libraries/**/*.cpp" + - "libraries/**/*.c" + - "libraries/**/*.h" + - "libraries/**/*.ino" + - "libraries/**/ci.json" + - "idf_component_examples/**" + - "idf_component.yml" + - "Kconfig.projbuild" + - "CMakeLists.txt" + - ".github/workflows/build_component.yml" + - ".github/scripts/check-cmakelists.sh" + - ".github/scripts/on-push-idf.sh" + - ".github/scripts/sketch_utils.sh" + - "variants/esp32/**" + - "variants/esp32c2/**" + - "variants/esp32c3/**" + - "variants/esp32c6/**" + - "variants/esp32h2/**" + - "variants/esp32p4/**" + - "variants/esp32s2/**" + - "variants/esp32s3/**" + - "!*.md" + - "!*.txt" + - "!*.properties" + +concurrency: + group: build-component-${{github.event.pull_request.number || github.ref}} + cancel-in-progress: true + +jobs: + cmake-check: + name: Check CMakeLists + runs-on: ubuntu-latest + if: ${{ !(github.event_name == 'pull_request' && startsWith(github.head_ref, 'release/')) }} + steps: + - uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683 # v4.2.2 + - run: bash ./.github/scripts/check-cmakelists.sh + + set-matrix: + name: Set Matrix + runs-on: ubuntu-latest + if: ${{ !(github.event_name == 'pull_request' && startsWith(github.head_ref, 'release/')) }} + outputs: + idf_ver: ${{ steps.set-matrix.outputs.idf_ver }} + idf_target: ${{ steps.set-matrix.outputs.idf_target }} + steps: + - name: Get IDF Version and Targets + id: set-matrix + run: | + # Default values + idf_ver="release-v5.3,release-v5.4,release-v5.5" + idf_targets="esp32,esp32s2,esp32s3,esp32c2,esp32c3,esp32c6,esp32h2,esp32p4" + + # Override with inputs if provided + if [[ -n "${{ inputs.idf_ver }}" ]]; then + idf_ver="${{ inputs.idf_ver }}" + fi + if [[ -n "${{ inputs.idf_targets }}" ]]; then + idf_targets="${{ inputs.idf_targets }}" + fi + + # Convert comma-separated strings to JSON arrays using a more robust method + idf_ver_json=$(printf '%s\n' "$idf_ver" | tr ',' '\n' | jq -R . | jq -s . | jq -c .) + idf_targets_json=$(printf '%s\n' "$idf_targets" | tr ',' '\n' | jq -R . | jq -s . | jq -c .) + + # Debug: Print the JSON for verification + echo "Debug - idf_ver_json: $idf_ver_json" + echo "Debug - idf_targets_json: $idf_targets_json" + + # Set outputs - ensure no extra whitespace + printf "idf_ver=%s\n" "$idf_ver_json" >> $GITHUB_OUTPUT + printf "idf_target=%s\n" "$idf_targets_json" >> $GITHUB_OUTPUT + + build-esp-idf-component: + name: Build IDF ${{ matrix.idf_ver }} for ${{ matrix.idf_target }} + runs-on: ubuntu-latest + needs: set-matrix + strategy: + fail-fast: false + matrix: + # The version names here correspond to the versions of espressif/idf Docker image. + # See https://hub.docker.com/r/espressif/idf/tags and + # https://docs.espressif.com/projects/esp-idf/en/latest/esp32/api-guides/tools/idf-docker-image.html + # for details. + idf_ver: ${{ fromJson(needs.set-matrix.outputs.idf_ver) }} + idf_target: ${{ fromJson(needs.set-matrix.outputs.idf_target) }} + container: espressif/idf:${{ matrix.idf_ver }} + steps: + - name: Check out arduino-esp32 as a component + uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683 # v4.2.2 + with: + submodules: recursive + path: components/arduino-esp32 + + - name: Setup jq + uses: dcarbone/install-jq-action@e397bd87438d72198f81efd21f876461183d383a # v3.0.1 + + - name: Build + env: + IDF_TARGET: ${{ matrix.idf_target }} + shell: bash + run: | + chmod a+x ./components/arduino-esp32/.github/scripts/* + ./components/arduino-esp32/.github/scripts/on-push-idf.sh + + - name: Upload generated sdkconfig files for debugging + uses: actions/upload-artifact@ea165f8d65b6e75b540449e92b4886f43607fa02 # v4.6.2 + if: always() + with: + name: sdkconfig-${{ matrix.idf_ver }}-${{ matrix.idf_target }} + path: ./components/arduino-esp32/idf_component_examples/**/sdkconfig diff --git a/.github/workflows/push.yml b/.github/workflows/push.yml index 1523b7231be..fc85566aa8b 100644 --- a/.github/workflows/push.yml +++ b/.github/workflows/push.yml @@ -25,34 +25,32 @@ on: pull_request: paths: - "cores/**" - - "libraries/**" - - "!libraries/**.md" - - "!libraries/**.txt" - - "!libraries/**.properties" - - "!libraries/**.py" + - "libraries/**/*.cpp" + - "libraries/**/*.c" + - "libraries/**/*.h" + - "libraries/**/*.ino" + - "libraries/**/ci.json" - "package/**" - - "idf_component_examples/**" - - "tools/**.py" + - "tools/get.*" - "platform.txt" - "programmers.txt" - - "idf_component.yml" - - "Kconfig.projbuild" - "package.json" - - "CMakeLists.txt" - ".github/workflows/push.yml" - - ".github/scripts/**" - - "!.github/scripts/find_*" - - "!.github/scripts/on-release.sh" - - "!.github/scripts/tests_*" - - "!.github/scripts/upload_*" - - "variants/esp32/**/*" - - "variants/esp32c3/**/*" - - "variants/esp32c5/**/*" - - "variants/esp32c6/**/*" - - "variants/esp32h2/**/*" - - "variants/esp32p4/**/*" - - "variants/esp32s2/**/*" - - "variants/esp32s3/**/*" + - ".github/scripts/install-*" + - ".github/scripts/on-push.sh" + - ".github/scripts/set_push_chunks.sh" + - ".github/scripts/sketch_utils.sh" + - "variants/esp32/**" + - "variants/esp32c3/**" + - "variants/esp32c5/**" + - "variants/esp32c6/**" + - "variants/esp32h2/**" + - "variants/esp32p4/**" + - "variants/esp32s2/**" + - "variants/esp32s3/**" + - "!*.md" + - "!*.txt" + - "!*.properties" concurrency: group: build-${{github.event.pull_request.number || github.ref}} @@ -62,14 +60,6 @@ env: MAX_CHUNKS: 15 jobs: - cmake-check: - name: Check cmake file - runs-on: ubuntu-latest - if: ${{ !(github.event_name == 'pull_request' && startsWith(github.head_ref, 'release/')) }} - steps: - - uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683 # v4.2.2 - - run: bash ./.github/scripts/check-cmakelists.sh - gen-chunks: name: Generate chunks runs-on: ubuntu-latest @@ -78,7 +68,6 @@ jobs: build_all: ${{ steps.set-chunks.outputs.build_all }} build_libraries: ${{ steps.set-chunks.outputs.build_libraries }} build_static_sketches: ${{ steps.set-chunks.outputs.build_static_sketches }} - build_idf: ${{ steps.set-chunks.outputs.build_idf }} chunk_count: ${{ steps.set-chunks.outputs.chunk_count }} chunks: ${{ steps.set-chunks.outputs.chunks }} steps: @@ -99,13 +88,7 @@ jobs: - 'tools/**' - 'platform.txt' - 'programmers.txt' - - "variants/esp32/**/*" - - "variants/esp32c3/**/*" - - "variants/esp32c6/**/*" - - "variants/esp32h2/**/*" - - "variants/esp32p4/**/*" - - "variants/esp32s2/**/*" - - "variants/esp32s3/**/*" + - "variants/**" libraries: - 'libraries/**/examples/**' - 'libraries/**/src/**' @@ -121,11 +104,6 @@ jobs: - 'libraries/NetworkClientSecure/src/**' - 'libraries/BLE/src/**' - 'libraries/Insights/src/**' - idf: - - 'idf_component.yml' - - 'Kconfig.projbuild' - - 'CMakeLists.txt' - - "idf_component_examples/**" - name: Set chunks id: set-chunks @@ -133,7 +111,6 @@ jobs: LIB_FILES: ${{ steps.changed-files.outputs.libraries_all_changed_files }} IS_PR: ${{ github.event_name == 'pull_request' }} MAX_CHUNKS: ${{ env.MAX_CHUNKS }} - BUILD_IDF: ${{ steps.changed-files.outputs.idf_any_changed == 'true' }} BUILD_LIBRARIES: ${{ steps.changed-files.outputs.libraries_any_changed == 'true' }} BUILD_STATIC_SKETCHES: ${{ steps.changed-files.outputs.static_sketeches_any_changed == 'true' }} FS_CHANGED: ${{ steps.changed-files.outputs.fs_any_changed == 'true' }} @@ -233,59 +210,6 @@ jobs: - name: Build Sketches run: bash ./.github/scripts/on-push.sh - build-esp-idf-component: - name: Build with ESP-IDF ${{ matrix.idf_ver }} for ${{ matrix.idf_target }} - needs: gen-chunks - if: | - needs.gen-chunks.outputs.build_all == 'true' || - needs.gen-chunks.outputs.build_libraries == 'true' || - needs.gen-chunks.outputs.build_idf == 'true' - runs-on: ubuntu-latest - strategy: - fail-fast: false - matrix: - # The version names here correspond to the versions of espressif/idf Docker image. - # See https://hub.docker.com/r/espressif/idf/tags and - # https://docs.espressif.com/projects/esp-idf/en/latest/esp32/api-guides/tools/idf-docker-image.html - # for details. - idf_ver: ["release-v5.3","release-v5.4","release-v5.5"] - idf_target: - [ - "esp32", - "esp32s2", - "esp32s3", - "esp32c2", - "esp32c3", - "esp32c6", - "esp32h2", - "esp32p4" - ] - container: espressif/idf:${{ matrix.idf_ver }} - steps: - - name: Check out arduino-esp32 as a component - uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683 # v4.2.2 - with: - submodules: recursive - path: components/arduino-esp32 - - - name: Setup jq - uses: dcarbone/install-jq-action@e397bd87438d72198f81efd21f876461183d383a # v3.0.1 - - - name: Build - env: - IDF_TARGET: ${{ matrix.idf_target }} - shell: bash - run: | - chmod a+x ./components/arduino-esp32/.github/scripts/* - ./components/arduino-esp32/.github/scripts/on-push-idf.sh - - - name: Upload generated sdkconfig files for debugging - uses: actions/upload-artifact@ea165f8d65b6e75b540449e92b4886f43607fa02 # v4.6.2 - if: always() - with: - name: sdkconfig-${{ matrix.idf_ver }}-${{ matrix.idf_target }} - path: ./components/arduino-esp32/idf_component_examples/**/sdkconfig - # Save artifacts to gh-pages save-master-artifacts: name: Save master artifacts diff --git a/.github/workflows/tests.yml b/.github/workflows/tests.yml index ddc9b64aace..058d9a3a793 100644 --- a/.github/workflows/tests.yml +++ b/.github/workflows/tests.yml @@ -16,20 +16,17 @@ on: pull_request: types: [opened, reopened, closed, synchronize, labeled, unlabeled] paths: + - ".github/scripts/install*.sh" - ".github/workflows/tests*" - - ".github/scripts/*.sh" - - "!.github/scripts/check-cmakelists.sh" - - "!.github/scripts/find_*" - - "!.github/scripts/on-*.sh" - - "!.github/scripts/set_push_chunks.sh" - - "!.github/scripts/update-version.sh" - - "!.github/scripts/upload_py_tools.sh" + - ".github/scripts/sketch_utils.sh" - "tests/**" - "cores/**" - "libraries/*/src/**.cpp" - "libraries/*/src/**.h" - "libraries/*/src/**.c" - "package/**" + - "!*.md" + - "!*.properties" schedule: - cron: "0 2 * * *" From b9c0f689ae3bdb7e40cc34b69ae2a8681a060af5 Mon Sep 17 00:00:00 2001 From: wulu Date: Wed, 30 Jul 2025 22:36:29 +0800 Subject: [PATCH 163/173] docs: Specify ADC resolution errata for ESP32-S2 v0.0 --- docs/en/api/adc.rst | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/docs/en/api/adc.rst b/docs/en/api/adc.rst index 416a8afd803..f2245cc1e5d 100644 --- a/docs/en/api/adc.rst +++ b/docs/en/api/adc.rst @@ -52,10 +52,12 @@ This function will return analog value in millivolts (calibrated). analogReadResolution ^^^^^^^^^^^^^^^^^^^^ -This function is used to set the resolution of ``analogRead`` return value. Default is 12 bits (range from 0 to 4095) -for all chips except ESP32-S2 where default is 13 bits (range from 0 to 8191). +This function is used to set the resolution of ``analogRead`` return value. Default is 12 bits (range from 0 to 4095) for all chips. When different resolution is set, the values read will be shifted to match the given resolution. +.. note:: + For **ESP32-S2 chip revision v0.0**, the default ADC resolution is 13 bits (0-8191) due to the `ADC-112 errata `_. This is fixed in later revisions (v1.0+), which use the standard 12-bit resolution. + Range is 1 - 16 .The default value will be used, if this function is not used. .. note:: For the ESP32, the resolution is between 9 to12 and it will change the ADC hardware resolution. Else value will be shifted. From 673048f73e41cfddccfdbc676f38f7ff7b146dfc Mon Sep 17 00:00:00 2001 From: Lucas Saavedra Vaz <32426024+lucasssvaz@users.noreply.github.com> Date: Thu, 31 Jul 2025 08:21:11 -0300 Subject: [PATCH 164/173] change(esptool): Upgrade esptool to 5.0.2 --- package/package_esp32_index.template.json | 60 +++++++++++------------ 1 file changed, 30 insertions(+), 30 deletions(-) diff --git a/package/package_esp32_index.template.json b/package/package_esp32_index.template.json index 3909c833c56..1b4b005dd34 100644 --- a/package/package_esp32_index.template.json +++ b/package/package_esp32_index.template.json @@ -81,7 +81,7 @@ { "packager": "esp32", "name": "esptool_py", - "version": "5.0.0" + "version": "5.0.2" }, { "packager": "esp32", @@ -469,56 +469,56 @@ }, { "name": "esptool_py", - "version": "5.0.0", + "version": "5.0.2", "systems": [ { "host": "aarch64-linux-gnu", - "url": "https://github.com/espressif/esptool/releases/download/v5.0.0/esptool-v5.0.0-linux-aarch64.tar.gz", - "archiveFileName": "esptool-v5.0.0-linux-aarch64.tar.gz", - "checksum": "SHA-256:2bf239f3ed76141a957cadb205b94414ec6da9ace4e85f285e247d20a92b83e3", - "size": "58231895" + "url": "https://github.com/espressif/esptool/releases/download/v5.0.2/esptool-v5.0.2-linux-aarch64.tar.gz", + "archiveFileName": "esptool-v5.0.2-linux-aarch64.tar.gz", + "checksum": "SHA-256:0c1fa4f5e96f715133fa65572063fa65bd120cd941b667cecd3360436a62b97b", + "size": "57815944" }, { "host": "x86_64-pc-linux-gnu", - "url": "https://github.com/espressif/esptool/releases/download/v5.0.0/esptool-v5.0.0-linux-amd64.tar.gz", - "archiveFileName": "esptool-v5.0.0-linux-amd64.tar.gz", - "checksum": "SHA-256:3b3835d266ac61f3242758f2fe34e3b33dbe6ee4b5acde005da793356f9f7043", - "size": "100783748" + "url": "https://github.com/espressif/esptool/releases/download/v5.0.2/esptool-v5.0.2-linux-amd64.tar.gz", + "archiveFileName": "esptool-v5.0.2-linux-amd64.tar.gz", + "checksum": "SHA-256:519e0015872d527bdca850b18575d4f635fa88ccdffe47a14c99a80a90b780c5", + "size": "100787554" }, { "host": "arm-linux-gnueabihf", - "url": "https://github.com/espressif/esptool/releases/download/v5.0.0/esptool-v5.0.0-linux-armv7.tar.gz", - "archiveFileName": "esptool-v5.0.0-linux-armv7.tar.gz", - "checksum": "SHA-256:e55cd321abecfcf27f72a2bff5d5e19a5365fd400de66d71c5e7218e77556315", - "size": "53461760" + "url": "https://github.com/espressif/esptool/releases/download/v5.0.2/esptool-v5.0.2-linux-armv7.tar.gz", + "archiveFileName": "esptool-v5.0.2-linux-armv7.tar.gz", + "checksum": "SHA-256:8df698d46a64b0b4a36d2a5bbd6bae58f81ca0a5e6451cd3f2d69a33340cc0f1", + "size": "53046401" }, { "host": "x86_64-apple-darwin", - "url": "https://github.com/espressif/esptool/releases/download/v5.0.0/esptool-v5.0.0-macos-amd64.tar.gz", - "archiveFileName": "esptool-v5.0.0-macos-amd64.tar.gz", - "checksum": "SHA-256:424da2bdf0435257ad81bcb7eae6fd8dd7f675ce5b2ee60032f4ecec4d6a5d45", - "size": "59629533" + "url": "https://github.com/espressif/esptool/releases/download/v5.0.2/esptool-v5.0.2-macos-amd64.tar.gz", + "archiveFileName": "esptool-v5.0.2-macos-amd64.tar.gz", + "checksum": "SHA-256:5c27295975515b97a9280f46845bd3acd5fc9e6a0583cfe03efa66cc50195ab0", + "size": "59619952" }, { "host": "arm64-apple-darwin", - "url": "https://github.com/espressif/esptool/releases/download/v5.0.0/esptool-v5.0.0-macos-arm64.tar.gz", - "archiveFileName": "esptool-v5.0.0-macos-arm64.tar.gz", - "checksum": "SHA-256:b91dfe1da7b0041376683dec10a91dfb266fbda2fb86ed87c4a034ff7182ee56", - "size": "56343104" + "url": "https://github.com/espressif/esptool/releases/download/v5.0.2/esptool-v5.0.2-macos-arm64.tar.gz", + "archiveFileName": "esptool-v5.0.2-macos-arm64.tar.gz", + "checksum": "SHA-256:93f0d9ef169f9bc6e32ed6f381977f63a5df7483b8002a5676dddf055bdbf775", + "size": "56344929" }, { "host": "x86_64-mingw32", - "url": "https://github.com/espressif/esptool/releases/download/v5.0.0/esptool-v5.0.0-windows-amd64.zip", - "archiveFileName": "esptool-v5.0.0-windows-amd64.zip", - "checksum": "SHA-256:2294107f66db6f09b886b337728a981173c9e7eab45a030928a8a5a1370611ca", - "size": "59105322" + "url": "https://github.com/espressif/esptool/releases/download/v5.0.2/esptool-v5.0.2-windows-amd64.zip", + "archiveFileName": "esptool-v5.0.2-windows-amd64.zip", + "checksum": "SHA-256:1caef993a16c5915714a0da772d93f2e3239f316c06223981b262b838287268c", + "size": "59097582" }, { "host": "i686-mingw32", - "url": "https://github.com/espressif/esptool/releases/download/v5.0.0/esptool-v5.0.0-windows-amd64.zip", - "archiveFileName": "esptool-v5.0.0-windows-amd64.zip", - "checksum": "SHA-256:2294107f66db6f09b886b337728a981173c9e7eab45a030928a8a5a1370611ca", - "size": "59105322" + "url": "https://github.com/espressif/esptool/releases/download/v5.0.2/esptool-v5.0.2-windows-amd64.zip", + "archiveFileName": "esptool-v5.0.2-windows-amd64.zip", + "checksum": "SHA-256:1caef993a16c5915714a0da772d93f2e3239f316c06223981b262b838287268c", + "size": "59097582" } ] }, From 30847588bc8909334479e4ee1ba48f43d051f8d0 Mon Sep 17 00:00:00 2001 From: Lucas Saavedra Vaz <32426024+lucasssvaz@users.noreply.github.com> Date: Thu, 31 Jul 2025 21:50:52 -0300 Subject: [PATCH 165/173] feat(codeql): Add CPP analysis --- .github/scripts/process_sarif.py | 129 +++++++++++++++++++++++++++++++ .github/workflows/codeql.yml | 55 ++++++++++++- 2 files changed, 183 insertions(+), 1 deletion(-) create mode 100644 .github/scripts/process_sarif.py diff --git a/.github/scripts/process_sarif.py b/.github/scripts/process_sarif.py new file mode 100644 index 00000000000..fad689b6390 --- /dev/null +++ b/.github/scripts/process_sarif.py @@ -0,0 +1,129 @@ +#!/usr/bin/env python3 + +# This script is used to process the SARIF file generated by CodeQL and +# to rename files back to .ino and adjust line numbers to match the original .ino files. + +import json +import sys +import os + +def process_artifact_location(artifact_location, renamed_files): + """ + Process a single artifact location to rename .cpp files back to .ino + """ + if 'uri' in artifact_location: + uri = artifact_location['uri'] + if uri in renamed_files: + print(f"Renaming file: {uri} -> {renamed_files[uri]}") + artifact_location['uri'] = renamed_files[uri] + return True + return False + +def process_region(region): + """ + Adjust line numbers in a region by decreasing them by 1 + """ + if 'startLine' in region: + region['startLine'] = max(1, region['startLine'] - 1) + if 'endLine' in region: + region['endLine'] = max(1, region['endLine'] - 1) + +def process_physical_location(physical_location, renamed_files): + """ + Process a physical location to rename files and adjust line numbers + """ + file_renamed = False + + if 'artifactLocation' in physical_location: + if process_artifact_location(physical_location['artifactLocation'], renamed_files): + file_renamed = True + + # Adjust line numbers if the file was renamed + if file_renamed and 'region' in physical_location: + process_region(physical_location['region']) + + return file_renamed + + +def process_sarif_file(sarif_file, renamed_files_file): + """ + Process SARIF file to rename files back to .ino and adjust line numbers + """ + # Read the renamed files mapping + with open(renamed_files_file, 'r') as f: + renamed_files = json.load(f) + + print(f"Loaded {len(renamed_files)} file mappings:") + for cpp_file, ino_file in renamed_files.items(): + print(f" {cpp_file} -> {ino_file}") + + + # Read the SARIF file + with open(sarif_file, 'r') as f: + sarif_data = json.load(f) + + files_processed = 0 + + # Process each run + if 'runs' in sarif_data: + for run in sarif_data['runs']: + # Process results + if 'results' in run: + for result in run['results']: + # Process all locations in the result + if 'locations' in result: + for location in result['locations']: + if 'physicalLocation' in location: + if process_physical_location(location['physicalLocation'], renamed_files): + files_processed += 1 + + # Process related locations if they exist + if 'relatedLocations' in result: + for location in result['relatedLocations']: + if 'physicalLocation' in location: + if process_physical_location(location['physicalLocation'], renamed_files): + files_processed += 1 + + # Process artifacts if they exist + if 'artifacts' in run: + for artifact in run['artifacts']: + if 'location' in artifact and 'uri' in artifact['location']: + uri = artifact['location']['uri'] + if uri in renamed_files: + artifact['location']['uri'] = renamed_files[uri] + files_processed += 1 + + print(f"Processed {files_processed} file references") + + # Write the processed SARIF file + with open(sarif_file, 'w') as f: + json.dump(sarif_data, f, indent=2) + +def main(): + if len(sys.argv) != 3: + print("Usage: python3 sarif_nobuild.py ") + sys.exit(1) + + sarif_file = sys.argv[1] + renamed_files_file = sys.argv[2] + + # Check if files exist + if not os.path.exists(sarif_file): + print(f"SARIF file not found: {sarif_file}") + sys.exit(1) + + if not os.path.exists(renamed_files_file): + print(f"Renamed files mapping not found: {renamed_files_file}") + sys.exit(1) + + try: + process_sarif_file(sarif_file, renamed_files_file) + print("SARIF file processed successfully") + except Exception as e: + print(f"Error processing SARIF file: {e}") + import traceback + traceback.print_exc() + sys.exit(1) + +if __name__ == "__main__": + main() diff --git a/.github/workflows/codeql.yml b/.github/workflows/codeql.yml index fb0badd78ec..9b2c6bccab1 100644 --- a/.github/workflows/codeql.yml +++ b/.github/workflows/codeql.yml @@ -7,6 +7,10 @@ on: - master pull_request: paths: + - "**/*.c" + - "**/*.cpp" + - "**/*.h" + - "**/*.ino" - "**/*.py" - ".github/workflows/*.yml" - ".github/workflows/*.yaml" @@ -17,15 +21,48 @@ jobs: runs-on: ubuntu-latest strategy: matrix: - language: [python, actions] + language: [python, actions, cpp] steps: - name: Checkout repository uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683 # v4.2.2 + - name: Process .ino files + if: matrix.language == 'cpp' + run: | + # Create a mapping file to track renamed files + echo "{}" > renamed_files.json + + # Find all .ino files and process them + find . -name "*.ino" -type f | while read -r file; do + echo "Processing $file" + + # Get the relative path from repository root + rel_path=$(realpath --relative-to=. "$file") + cpp_path="${rel_path%.ino}.cpp" + + # Create new .cpp file with Arduino.h include + echo "#include " > "$cpp_path" + + # Append the original content + cat "$file" >> "$cpp_path" + + # Update the mapping file + jq --arg ino "$rel_path" --arg cpp "$cpp_path" '. += {($cpp): $ino}' renamed_files.json > temp.json && mv temp.json renamed_files.json + + # Remove the original .ino file + rm "$file" + + echo "Converted $file to $cpp_path" + done + + echo "Renamed files mapping:" + cat renamed_files.json + - name: Initialize CodeQL uses: github/codeql-action/init@181d5eefc20863364f96762470ba6f862bdef56b # v3.29.2 with: + build-mode: none languages: ${{ matrix.language }} config-file: ./.github/codeql/codeql-config.yml @@ -33,3 +70,19 @@ jobs: uses: github/codeql-action/analyze@181d5eefc20863364f96762470ba6f862bdef56b # v3.29.2 with: category: "/language:${{ matrix.language }}" + output: sarif-results + upload: failure-only + + - name: Process SARIF file + if: matrix.language == 'cpp' + run: | + sarif_file="sarif-results/${{ matrix.language }}.sarif" + + # Run the Python script to process the SARIF file + python3 .github/scripts/process_sarif.py "$sarif_file" "renamed_files.json" + + - name: Upload SARIF file + uses: github/codeql-action/upload-sarif@181d5eefc20863364f96762470ba6f862bdef56b # v3.29.2 + with: + sarif_file: sarif-results/${{ matrix.language }}.sarif + category: "/language:${{ matrix.language }}" From 81cc594e63f50d5f3c2045b0cb069f06ff46fc27 Mon Sep 17 00:00:00 2001 From: Lucas Saavedra Vaz <32426024+lucasssvaz@users.noreply.github.com> Date: Sun, 3 Aug 2025 17:32:18 -0300 Subject: [PATCH 166/173] fix(push): Fix typo in yaml --- .github/workflows/push.yml | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/.github/workflows/push.yml b/.github/workflows/push.yml index fc85566aa8b..48530e30bc9 100644 --- a/.github/workflows/push.yml +++ b/.github/workflows/push.yml @@ -96,7 +96,7 @@ jobs: - 'libraries/Network/src/**' fs: - 'libraries/FS/src/**' - static_sketeches: + static_sketches: - 'libraries/NetworkClientSecure/examples/WiFiClientSecure/WiFiClientSecure.ino' - 'libraries/BLE/examples/Server/Server.ino' - 'libraries/ESP32/examples/Camera/CameraWebServer/CameraWebServer.ino' @@ -112,7 +112,7 @@ jobs: IS_PR: ${{ github.event_name == 'pull_request' }} MAX_CHUNKS: ${{ env.MAX_CHUNKS }} BUILD_LIBRARIES: ${{ steps.changed-files.outputs.libraries_any_changed == 'true' }} - BUILD_STATIC_SKETCHES: ${{ steps.changed-files.outputs.static_sketeches_any_changed == 'true' }} + BUILD_STATIC_SKETCHES: ${{ steps.changed-files.outputs.static_sketches_any_changed == 'true' }} FS_CHANGED: ${{ steps.changed-files.outputs.fs_any_changed == 'true' }} NETWORKING_CHANGED: ${{ steps.changed-files.outputs.networking_any_changed == 'true' }} CORE_CHANGED: ${{ steps.changed-files.outputs.core_any_changed == 'true' }} From f6b1910e8ba5a9aff99fa00b2133b10f34f85e4f Mon Sep 17 00:00:00 2001 From: Lucas Saavedra Vaz <32426024+lucasssvaz@users.noreply.github.com> Date: Fri, 8 Aug 2025 10:22:19 -0300 Subject: [PATCH 167/173] fix(codeql): Remove rule causing false positives --- .github/codeql/codeql-config.yml | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/.github/codeql/codeql-config.yml b/.github/codeql/codeql-config.yml index c640d7bd6ef..1c284c9d68e 100644 --- a/.github/codeql/codeql-config.yml +++ b/.github/codeql/codeql-config.yml @@ -20,7 +20,9 @@ query-filters: problem.severity: - recommendation - exclude: - id: tob/cpp/use-of-legacy-algorithm + id: tob/cpp/use-of-legacy-algorithm # We use legacy algorithms in many places for integrity checks + - exclude: + id: cpp/dead-code-goto # Too many false positives in no-build mode paths-ignore: - tests/** From 49e1676ae28470bd8168fc5ca03a80462353e8f5 Mon Sep 17 00:00:00 2001 From: daniel watson <165518737+ozzloy@users.noreply.github.com> Date: Fri, 8 Aug 2025 13:03:22 -0700 Subject: [PATCH 168/173] put valid category for ESP_I2S ESP_I2S was categorized as "Sound", which is not a valid category. "Signal Input/Output" is a valid category. see https://arduino.github.io/arduino-cli/1.2/library-specification/#libraryproperties-file-format for a list of valid categories (search for "category" on that page). current valid categories are: - Display - Communication - Signal Input/Output - Sensors - Device Control - Timing - Data Storage - Data Processing - Other --- libraries/ESP_I2S/library.properties | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/libraries/ESP_I2S/library.properties b/libraries/ESP_I2S/library.properties index b2763f4e7e8..18c48c095a8 100644 --- a/libraries/ESP_I2S/library.properties +++ b/libraries/ESP_I2S/library.properties @@ -4,6 +4,6 @@ author=me-no-dev maintainer=me-no-dev sentence=Library for ESP I2S communication paragraph=Supports ESP32 Arduino platforms. -category=Sound +category=Signal Input/Output url=https://github.com/espressif/arduino-esp32/ architectures=esp32 From 9d13bcc674bb7176df5d52f319a6cb1b16e17653 Mon Sep 17 00:00:00 2001 From: Lucas Saavedra Vaz <32426024+lucasssvaz@users.noreply.github.com> Date: Wed, 13 Aug 2025 22:15:41 -0300 Subject: [PATCH 169/173] fix(tests): Change results folder --- .github/PULL_REQUEST_TEMPLATE.md | 2 +- .github/workflows/tests.yml | 36 ++++++++++++++++++++++++++++- .github/workflows/tests_build.yml | 6 ++--- .github/workflows/tests_hw.yml | 31 +++++++++++++++++++++---- .github/workflows/tests_qemu.yml | 8 +++---- .github/workflows/tests_results.yml | 12 +++++----- .github/workflows/tests_wokwi.yml | 22 +++++++++++++----- README.md | 2 +- 8 files changed, 93 insertions(+), 26 deletions(-) diff --git a/.github/PULL_REQUEST_TEMPLATE.md b/.github/PULL_REQUEST_TEMPLATE.md index 5e0d99c0457..6a803943690 100644 --- a/.github/PULL_REQUEST_TEMPLATE.md +++ b/.github/PULL_REQUEST_TEMPLATE.md @@ -13,7 +13,7 @@ ## Description of Change Please describe your proposed Pull Request and it's impact. -## Tests scenarios +## Test Scenarios Please describe on what Hardware and Software combinations you have tested this Pull Request and how. (*eg. I have tested my Pull Request on Arduino-esp32 core v2.0.2 with ESP32 and ESP32-S2 Board with this scenario*) diff --git a/.github/workflows/tests.yml b/.github/workflows/tests.yml index 058d9a3a793..925855844ac 100644 --- a/.github/workflows/tests.yml +++ b/.github/workflows/tests.yml @@ -28,7 +28,7 @@ on: - "!*.md" - "!*.properties" schedule: - - cron: "0 2 * * *" + - cron: "0 0 * * *" concurrency: group: tests-${{ github.event.pull_request.number || github.ref }} @@ -45,9 +45,43 @@ jobs: name: event_file path: ${{ github.event_path }} + check-last-run: + name: Check last run + runs-on: ubuntu-latest + outputs: + run_tests: ${{ steps.last-run.outputs.run_tests }} + steps: + - name: Checkout + if: github.event_name != 'pull_request' + # if: github.event_name == 'schedule' + uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683 # v4.2.2 + with: + ref: gh-pages + + - name: Get last run commit and compare with current commit + id: last-run + run: | + if [ ${{ github.event_name }} != 'pull_request' ]; then + # if [ ${{ github.event_name }} == 'schedule' ]; then + last_run_commit=$(jq -r '.commit_sha' ./runtime-test-results/test_results.json) + current_commit=${{ github.sha }} + if [ "$last_run_commit" == "$current_commit" ]; then + echo "Last run commit and current commit are the same. Skipping tests..." + echo "run_tests=false" >> $GITHUB_OUTPUT + else + echo "Last run commit and current commit are different. Running tests..." + echo "run_tests=true" >> $GITHUB_OUTPUT + fi + else + echo "Not a schedule event. Continuing..." + echo "run_tests=true" >> $GITHUB_OUTPUT + fi + gen-matrix: name: Generate matrix runs-on: ubuntu-latest + if: ${{ needs.check-last-run.outputs.run_tests }} + needs: check-last-run outputs: build-types: ${{ steps.set-matrix.outputs.build-types }} hw-types: ${{ steps.set-matrix.outputs.hw-types }} diff --git a/.github/workflows/tests_build.yml b/.github/workflows/tests_build.yml index ac1f40644ed..c74224941af 100644 --- a/.github/workflows/tests_build.yml +++ b/.github/workflows/tests_build.yml @@ -24,7 +24,7 @@ jobs: if: github.event.pull_request.number != null uses: actions/cache/restore@5a3ec84eff668545956fd18022155c47e93e2684 # v4.2.3 with: - key: tests-${{ env.id }}-bin + key: test-${{ env.id }}-bin path: | ~/.arduino/tests/${{ inputs.chip }}/**/build*.tmp/*.bin ~/.arduino/tests/${{ inputs.chip }}/**/build*.tmp/*.elf @@ -71,7 +71,7 @@ jobs: uses: actions/cache/save@5a3ec84eff668545956fd18022155c47e93e2684 # v4.2.3 if: steps.check-build.outputs.enabled == 'true' && github.event.pull_request.number != null with: - key: tests-${{ env.id }}-bin + key: test-${{ env.id }}-bin path: | ~/.arduino/tests/${{ inputs.chip }}/**/build*.tmp/*.bin ~/.arduino/tests/${{ inputs.chip }}/**/build*.tmp/*.elf @@ -81,7 +81,7 @@ jobs: - name: Upload ${{ inputs.chip }} ${{ inputs.type }} binaries as artifacts uses: actions/upload-artifact@ea165f8d65b6e75b540449e92b4886f43607fa02 # v4.6.2 with: - name: tests-bin-${{ inputs.chip }}-${{ inputs.type }} + name: test-bin-${{ inputs.chip }}-${{ inputs.type }} overwrite: true path: | ~/.arduino/tests/${{ inputs.chip }}/**/build*.tmp/*.bin diff --git a/.github/workflows/tests_hw.yml b/.github/workflows/tests_hw.yml index 6f5fc67f7b9..d9fb755c94a 100644 --- a/.github/workflows/tests_hw.yml +++ b/.github/workflows/tests_hw.yml @@ -39,7 +39,7 @@ jobs: if: github.event.pull_request.number != null uses: actions/cache/restore@5a3ec84eff668545956fd18022155c47e93e2684 # v4.2.3 with: - key: tests-${{ env.id }}-results-hw + key: test-${{ env.id }}-results-hw path: | tests/**/*.xml tests/**/result_*.json @@ -61,6 +61,7 @@ jobs: if: ${{ steps.check-tests.outputs.enabled == 'true' }} uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683 # v4.2.2 with: + # Workaround for missing files in checkout sparse-checkout: | * @@ -84,7 +85,7 @@ jobs: if: ${{ steps.check-tests.outputs.enabled == 'true' }} uses: actions/download-artifact@95815c38cf2ff2164869cbab79da8d1f422bc89e # v4.2.1 with: - name: tests-bin-${{ inputs.chip }}-${{ inputs.type }} + name: test-bin-${{ inputs.chip }}-${{ inputs.type }} path: | ~/.arduino/tests/${{ inputs.chip }} @@ -102,7 +103,7 @@ jobs: uses: actions/cache/save@5a3ec84eff668545956fd18022155c47e93e2684 # v4.2.3 if: steps.check-tests.outputs.enabled == 'true' && github.event.pull_request.number != null with: - key: tests-${{ env.id }}-results-hw + key: test-${{ env.id }}-results-hw path: | tests/**/*.xml tests/**/result_*.json @@ -111,7 +112,29 @@ jobs: uses: actions/upload-artifact@ea165f8d65b6e75b540449e92b4886f43607fa02 # v4.6.2 if: always() with: - name: tests-results-hw-${{ inputs.chip }}-${{ inputs.type }} + name: test-results-hw-${{ inputs.chip }}-${{ inputs.type }} + overwrite: true + path: | + tests/**/*.xml + tests/**/result_*.json + + gitlab-tests: + name: Internal Hardware tests + runs-on: ubuntu-latest + if: false + # if: github.event_name == 'schedule' + steps: + # The decision to run the tests will be made inside GitLab + # GitLab should only perform the actual test execution and save the results as artifacts in + # a format that can be downloaded and parsed by this GitHub Actions workflow. + + + + - name: Upload Internal Hardware test results as artifacts + uses: actions/upload-artifact@ea165f8d65b6e75b540449e92b4886f43607fa02 # v4.6.2 + if: always() + with: + name: test-results-gitlab overwrite: true path: | tests/**/*.xml diff --git a/.github/workflows/tests_qemu.yml b/.github/workflows/tests_qemu.yml index fa3f874cbbb..d9ddf6e1b0c 100644 --- a/.github/workflows/tests_qemu.yml +++ b/.github/workflows/tests_qemu.yml @@ -23,7 +23,7 @@ jobs: if: github.event.pull_request.number != null uses: actions/cache/restore@5a3ec84eff668545956fd18022155c47e93e2684 # v4.2.3 with: - key: tests-${{ env.id }}-results-qemu + key: test-${{ env.id }}-results-qemu path: | tests/**/*.xml tests/**/result_*.json @@ -115,7 +115,7 @@ jobs: if: ${{ steps.check-tests.outputs.enabled == 'true' }} uses: actions/download-artifact@95815c38cf2ff2164869cbab79da8d1f422bc89e # v4.2.1 with: - name: tests-bin-${{ inputs.chip }}-${{ inputs.type }} + name: test-bin-${{ inputs.chip }}-${{ inputs.type }} path: | ~/.arduino/tests/${{ inputs.chip }} @@ -127,7 +127,7 @@ jobs: uses: actions/cache/save@5a3ec84eff668545956fd18022155c47e93e2684 # v4.2.3 if: steps.check-tests.outputs.enabled == 'true' && github.event.pull_request.number != null with: - key: tests-${{ env.id }}-results-qemu + key: test-${{ env.id }}-results-qemu path: | tests/**/*.xml tests/**/result_*.json @@ -136,7 +136,7 @@ jobs: uses: actions/upload-artifact@ea165f8d65b6e75b540449e92b4886f43607fa02 # v4.6.2 if: always() with: - name: tests-results-qemu-${{ inputs.chip }}-${{ inputs.type }} + name: test-results-qemu-${{ inputs.chip }}-${{ inputs.type }} overwrite: true path: | tests/**/*.xml diff --git a/.github/workflows/tests_results.yml b/.github/workflows/tests_results.yml index ebba2a3aa08..28227bfd3b1 100644 --- a/.github/workflows/tests_results.yml +++ b/.github/workflows/tests_results.yml @@ -113,7 +113,7 @@ jobs: with: script: | const ref = process.env.original_ref; - const key_prefix = 'tests-' + ref + '-'; + const key_prefix = 'test-' + ref + '-'; if (process.env.original_event == 'pull_request' && process.env.original_action != 'closed') { console.log('Skipping cache cleanup for open PR'); @@ -163,15 +163,15 @@ jobs: - name: Generate report if: ${{ !cancelled() && (env.original_event == 'schedule' || env.original_event == 'workflow_dispatch') }} # codespell:ignore cancelled env: - REPORT_FILE: ./runtime-tests-results/RUNTIME_TESTS_REPORT.md + REPORT_FILE: ./runtime-test-results/RUNTIME_TESTS_REPORT.md WOKWI_RUN_ID: ${{ github.event.workflow_run.id }} BUILD_RUN_ID: ${{ env.original_run_id }} IS_FAILING: ${{ env.original_conclusion == 'failure' || env.original_conclusion == 'timed_out' || github.event.workflow_run.conclusion == 'failure' || github.event.workflow_run.conclusion == 'timed_out' || job.status == 'failure' }} run: | rm -rf artifacts $REPORT_FILE - mv -f ./unity_results.json ./runtime-tests-results/unity_results.json + mv -f ./unity_results.json ./runtime-test-results/unity_results.json touch $REPORT_FILE - python3 ./runtime-tests-results/table_generator.py ./runtime-tests-results/unity_results.json >> $REPORT_FILE + python3 ./runtime-test-results/table_generator.py ./runtime-test-results/unity_results.json >> $REPORT_FILE - name: Generate badge if: ${{ !cancelled() && (env.original_event == 'schedule' || env.original_event == 'workflow_dispatch') }} # codespell:ignore cancelled @@ -179,7 +179,7 @@ jobs: with: label: Runtime Tests status: ${{ job.status == 'success' && 'passing' || 'failing' }} - output: runtime-tests-results/badge.svg + output: runtime-test-results/badge.svg color: ${{ job.status == 'success' && 'green' || 'red' }} style: flat @@ -190,6 +190,6 @@ jobs: git config user.email "41898282+github-actions[bot]@users.noreply.github.com" if [[ `git status --porcelain` ]]; then git add --all - git commit -m "Updated runtime tests report" + git commit -m "Updated runtime test results" git push origin HEAD:gh-pages fi diff --git a/.github/workflows/tests_wokwi.yml b/.github/workflows/tests_wokwi.yml index 03dd64fc0fb..6af66cf0e42 100644 --- a/.github/workflows/tests_wokwi.yml +++ b/.github/workflows/tests_wokwi.yml @@ -135,17 +135,27 @@ jobs: with: github-token: ${{ secrets.GITHUB_TOKEN }} run-id: ${{ github.event.workflow_run.id }} - pattern: tests-results-hw-* + pattern: test-results-hw-* merge-multiple: true path: artifacts/results/hw + - name: Download and extract parent GitLab results + uses: actions/download-artifact@95815c38cf2ff2164869cbab79da8d1f422bc89e # v4.2.1 + continue-on-error: true + with: + github-token: ${{ secrets.GITHUB_TOKEN }} + run-id: ${{ github.event.workflow_run.id }} + pattern: test-results-gitlab + merge-multiple: true + path: artifacts/results/gitlab + - name: Download and extract parent QEMU results uses: actions/download-artifact@95815c38cf2ff2164869cbab79da8d1f422bc89e # v4.2.1 continue-on-error: true with: github-token: ${{ secrets.GITHUB_TOKEN }} run-id: ${{ github.event.workflow_run.id }} - pattern: tests-results-qemu-* + pattern: test-results-qemu-* merge-multiple: true path: artifacts/results/qemu @@ -221,7 +231,7 @@ jobs: if: needs.get-artifacts.outputs.pr_num uses: actions/cache/restore@5a3ec84eff668545956fd18022155c47e93e2684 # v4.2.3 with: - key: tests-${{ env.id }}-results-wokwi + key: test-${{ env.id }}-results-wokwi path: | tests/**/*.xml tests/**/result_*.json @@ -274,7 +284,7 @@ jobs: with: github-token: ${{ secrets.GITHUB_TOKEN }} run-id: ${{ github.event.workflow_run.id }} - name: tests-bin-${{ matrix.chip }}-${{ matrix.type }} + name: test-bin-${{ matrix.chip }}-${{ matrix.type }} path: | ~/.arduino/tests/${{ matrix.chip }} @@ -289,7 +299,7 @@ jobs: uses: actions/cache/save@5a3ec84eff668545956fd18022155c47e93e2684 # v4.2.3 if: steps.check-tests.outputs.enabled == 'true' && needs.get-artifacts.outputs.pr_num with: - key: tests-${{ env.id }}-results-wokwi + key: test-${{ env.id }}-results-wokwi path: | tests/**/*.xml tests/**/result_*.json @@ -298,7 +308,7 @@ jobs: uses: actions/upload-artifact@ea165f8d65b6e75b540449e92b4886f43607fa02 # v4.6.2 if: always() with: - name: tests-results-wokwi-${{ matrix.chip }}-${{ matrix.type }} + name: test-results-wokwi-${{ matrix.chip }}-${{ matrix.type }} overwrite: true path: | tests/**/*.xml diff --git a/README.md b/README.md index f40315c03cc..f8ff1f28eaa 100644 --- a/README.md +++ b/README.md @@ -3,7 +3,7 @@ [![Build Status](https://img.shields.io/github/actions/workflow/status/espressif/arduino-esp32/push.yml?branch=master&event=push&label=Compilation%20Tests)](https://github.com/espressif/arduino-esp32/actions/workflows/push.yml?query=branch%3Amaster+event%3Apush) [![Verbose Build Status](https://img.shields.io/github/actions/workflow/status/espressif/arduino-esp32/push.yml?branch=master&event=schedule&label=Compilation%20Tests%20(Verbose))](https://github.com/espressif/arduino-esp32/actions/workflows/push.yml?query=branch%3Amaster+event%3Aschedule) [![External Libraries Test](https://img.shields.io/github/actions/workflow/status/espressif/arduino-esp32/lib.yml?branch=master&event=schedule&label=External%20Libraries%20Test)](https://github.com/espressif/arduino-esp32/blob/gh-pages/LIBRARIES_TEST.md) -[![Runtime Tests](https://github.com/espressif/arduino-esp32/blob/gh-pages/runtime-tests-results/badge.svg)](https://github.com/espressif/arduino-esp32/blob/gh-pages/runtime-tests-results/RUNTIME_TESTS_REPORT.md) +[![Runtime Tests](https://github.com/espressif/arduino-esp32/blob/gh-pages/runtime-test-results/badge.svg)](https://github.com/espressif/arduino-esp32/blob/gh-pages/runtime-test-results/RUNTIME_TEST_RESULTS.md) ### Need help or have a question? Join the chat at [Discord](https://discord.gg/8xY6e9crwv) or [open a new Discussion](https://github.com/espressif/arduino-esp32/discussions) From e38dba808f2a189e53d2e683f2d0faa8085e7605 Mon Sep 17 00:00:00 2001 From: Lucas Saavedra Vaz <32426024+lucasssvaz@users.noreply.github.com> Date: Wed, 13 Aug 2025 22:22:53 -0300 Subject: [PATCH 170/173] fix(tests): Fix last commit check --- .github/workflows/tests.yml | 12 +++++++++--- 1 file changed, 9 insertions(+), 3 deletions(-) diff --git a/.github/workflows/tests.yml b/.github/workflows/tests.yml index 925855844ac..4b915cf36de 100644 --- a/.github/workflows/tests.yml +++ b/.github/workflows/tests.yml @@ -63,13 +63,19 @@ jobs: run: | if [ ${{ github.event_name }} != 'pull_request' ]; then # if [ ${{ github.event_name }} == 'schedule' ]; then - last_run_commit=$(jq -r '.commit_sha' ./runtime-test-results/test_results.json) + if [ -f ./runtime-test-results/test_results.json ]; then + last_run_commit=$(jq -r '.commit_sha' ./runtime-test-results/test_results.json) + last_run_failed=$(jq -r '.tests_failed' ./runtime-test-results/test_results.json) + else + last_run_commit="none" + last_run_failed="true" + fi current_commit=${{ github.sha }} - if [ "$last_run_commit" == "$current_commit" ]; then + if [ "$last_run_commit" == "$current_commit" ] && [ "$last_run_failed" == "false" ]; then echo "Last run commit and current commit are the same. Skipping tests..." echo "run_tests=false" >> $GITHUB_OUTPUT else - echo "Last run commit and current commit are different. Running tests..." + echo "Last run commit and current commit are different or last run failed. Running tests..." echo "run_tests=true" >> $GITHUB_OUTPUT fi else From ead6c1cc9485d1f68d369c643c3efd8358f2a283 Mon Sep 17 00:00:00 2001 From: Lucas Saavedra Vaz <32426024+lucasssvaz@users.noreply.github.com> Date: Wed, 13 Aug 2025 23:52:27 -0300 Subject: [PATCH 171/173] fix(tests): Move results to proper folder --- .github/workflows/tests_hw.yml | 22 ---------------------- .github/workflows/tests_results.yml | 1 + 2 files changed, 1 insertion(+), 22 deletions(-) diff --git a/.github/workflows/tests_hw.yml b/.github/workflows/tests_hw.yml index d9fb755c94a..e095ebe6dc8 100644 --- a/.github/workflows/tests_hw.yml +++ b/.github/workflows/tests_hw.yml @@ -117,25 +117,3 @@ jobs: path: | tests/**/*.xml tests/**/result_*.json - - gitlab-tests: - name: Internal Hardware tests - runs-on: ubuntu-latest - if: false - # if: github.event_name == 'schedule' - steps: - # The decision to run the tests will be made inside GitLab - # GitLab should only perform the actual test execution and save the results as artifacts in - # a format that can be downloaded and parsed by this GitHub Actions workflow. - - - - - name: Upload Internal Hardware test results as artifacts - uses: actions/upload-artifact@ea165f8d65b6e75b540449e92b4886f43607fa02 # v4.6.2 - if: always() - with: - name: test-results-gitlab - overwrite: true - path: | - tests/**/*.xml - tests/**/result_*.json diff --git a/.github/workflows/tests_results.yml b/.github/workflows/tests_results.yml index 28227bfd3b1..56883ecf52e 100644 --- a/.github/workflows/tests_results.yml +++ b/.github/workflows/tests_results.yml @@ -172,6 +172,7 @@ jobs: mv -f ./unity_results.json ./runtime-test-results/unity_results.json touch $REPORT_FILE python3 ./runtime-test-results/table_generator.py ./runtime-test-results/unity_results.json >> $REPORT_FILE + mv -f ./test_results.json ./runtime-test-results/test_results.json - name: Generate badge if: ${{ !cancelled() && (env.original_event == 'schedule' || env.original_event == 'workflow_dispatch') }} # codespell:ignore cancelled From faa9388d203071f63ef75c59875381623ca5ed36 Mon Sep 17 00:00:00 2001 From: Lucas Saavedra Vaz <32426024+lucasssvaz@users.noreply.github.com> Date: Thu, 14 Aug 2025 09:23:24 -0300 Subject: [PATCH 172/173] fix(tests): Remove useless check --- .github/workflows/tests.yml | 40 ----------------------------- .github/workflows/tests_results.yml | 2 +- 2 files changed, 1 insertion(+), 41 deletions(-) diff --git a/.github/workflows/tests.yml b/.github/workflows/tests.yml index 4b915cf36de..bd41f2ec61f 100644 --- a/.github/workflows/tests.yml +++ b/.github/workflows/tests.yml @@ -45,49 +45,9 @@ jobs: name: event_file path: ${{ github.event_path }} - check-last-run: - name: Check last run - runs-on: ubuntu-latest - outputs: - run_tests: ${{ steps.last-run.outputs.run_tests }} - steps: - - name: Checkout - if: github.event_name != 'pull_request' - # if: github.event_name == 'schedule' - uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683 # v4.2.2 - with: - ref: gh-pages - - - name: Get last run commit and compare with current commit - id: last-run - run: | - if [ ${{ github.event_name }} != 'pull_request' ]; then - # if [ ${{ github.event_name }} == 'schedule' ]; then - if [ -f ./runtime-test-results/test_results.json ]; then - last_run_commit=$(jq -r '.commit_sha' ./runtime-test-results/test_results.json) - last_run_failed=$(jq -r '.tests_failed' ./runtime-test-results/test_results.json) - else - last_run_commit="none" - last_run_failed="true" - fi - current_commit=${{ github.sha }} - if [ "$last_run_commit" == "$current_commit" ] && [ "$last_run_failed" == "false" ]; then - echo "Last run commit and current commit are the same. Skipping tests..." - echo "run_tests=false" >> $GITHUB_OUTPUT - else - echo "Last run commit and current commit are different or last run failed. Running tests..." - echo "run_tests=true" >> $GITHUB_OUTPUT - fi - else - echo "Not a schedule event. Continuing..." - echo "run_tests=true" >> $GITHUB_OUTPUT - fi - gen-matrix: name: Generate matrix runs-on: ubuntu-latest - if: ${{ needs.check-last-run.outputs.run_tests }} - needs: check-last-run outputs: build-types: ${{ steps.set-matrix.outputs.build-types }} hw-types: ${{ steps.set-matrix.outputs.hw-types }} diff --git a/.github/workflows/tests_results.yml b/.github/workflows/tests_results.yml index 56883ecf52e..d33221c129d 100644 --- a/.github/workflows/tests_results.yml +++ b/.github/workflows/tests_results.yml @@ -163,7 +163,7 @@ jobs: - name: Generate report if: ${{ !cancelled() && (env.original_event == 'schedule' || env.original_event == 'workflow_dispatch') }} # codespell:ignore cancelled env: - REPORT_FILE: ./runtime-test-results/RUNTIME_TESTS_REPORT.md + REPORT_FILE: ./runtime-test-results/RUNTIME_TEST_RESULTS.md WOKWI_RUN_ID: ${{ github.event.workflow_run.id }} BUILD_RUN_ID: ${{ env.original_run_id }} IS_FAILING: ${{ env.original_conclusion == 'failure' || env.original_conclusion == 'timed_out' || github.event.workflow_run.conclusion == 'failure' || github.event.workflow_run.conclusion == 'timed_out' || job.status == 'failure' }} From b3b65b1cb5b8316d3edb8146023c8f62ef19b54a Mon Sep 17 00:00:00 2001 From: wulu Date: Thu, 14 Aug 2025 22:20:09 +0800 Subject: [PATCH 173/173] fix(docs): Correct list indentation for analogWrite parameter --- docs/en/api/ledc.rst | 1 + 1 file changed, 1 insertion(+) diff --git a/docs/en/api/ledc.rst b/docs/en/api/ledc.rst index 6ea3437bbf5..d3d6da1e0f6 100644 --- a/docs/en/api/ledc.rst +++ b/docs/en/api/ledc.rst @@ -296,6 +296,7 @@ It is compatible with Arduinos analogWrite function. * ``pin`` select the GPIO pin. * ``value`` select the duty cycle of pwm. + * range is from 0 (always off) to 255 (always on). analogWriteResolution

*ptM@WNYc*CowO^YW?Cx0Nz9(wGHZ8tH; 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 p-}>{(!?V9VeR=xcqZxe( +#include +#include +#include +#include +#include +#ifdef DISPLAY_ENABLED +#define SCREEN_WIDTH 128 +#define SCREEN_HEIGHT 64 +#define OLED_RESET -1 +#include +#include +Adafruit_SSD1306 display(SCREEN_WIDTH, SCREEN_HEIGHT, &Wire, OLED_RESET); +#include +Adafruit_NeoPixel rgb_led_1 = Adafruit_NeoPixel(1, 1, NEO_GRB + NEO_KHZ800); + +#endif +#include "esp_partition.h" +#include "esp_ota_ops.h" +#include "esp_system.h" + +String ssid; +uint8_t mac[6]; + +// Create an instance of the server +WebServer server(80); +bool displayEnabled; + +const int BUTTON_PIN = 0; // GPIO for the button +volatile unsigned long lastPressTime = 0; // Time of last button press +volatile bool doublePressDetected = false; // Flag for double press +const unsigned long doublePressInterval = 500; // Max. time (in ms) between two presses for double press +volatile int pressCount = 0; // Counts the button presses + +const unsigned char epd_bitmap_wifi[] PROGMEM = { + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x1f, 0x80, 0x00, 0x00, 0x00, 0x01, 0xff, 0xf0, 0x00, 0x00, 0x00, 0x07, 0xff, 0xfc, 0x00, 0x00, 0x00, 0x1f, 0xe0, 0xff, 0x00, 0x00, + 0x00, 0x3f, 0x00, 0x0f, 0x80, 0x00, 0x00, 0x7c, 0x00, 0x03, 0xe0, 0x00, 0x00, 0xf0, 0x00, 0x01, 0xf0, 0x00, 0x01, 0xe0, 0x00, 0x00, 0x78, 0x00, + 0x03, 0xc0, 0x00, 0x00, 0x38, 0x00, 0x07, 0x80, 0x00, 0x00, 0x1c, 0x00, 0x0f, 0x00, 0x06, 0x00, 0x0e, 0x00, 0x0e, 0x00, 0x7f, 0xe0, 0x0e, 0x00, + 0x0c, 0x01, 0xff, 0xf0, 0x06, 0x00, 0x00, 0x07, 0xff, 0xfc, 0x02, 0x00, 0x00, 0x0f, 0x80, 0x3e, 0x00, 0x00, 0x00, 0x1f, 0x00, 0x0f, 0x00, 0x00, + 0x00, 0x1c, 0x00, 0x07, 0x80, 0x00, 0x00, 0x38, 0x00, 0x03, 0xc0, 0x00, 0x00, 0x70, 0x00, 0x01, 0xc0, 0x00, 0x00, 0x70, 0x00, 0x00, 0xc0, 0x00, + 0x00, 0x20, 0x00, 0x00, 0x40, 0x00, 0x00, 0x00, 0x1f, 0x80, 0x00, 0x00, 0x00, 0x00, 0x7f, 0xc0, 0x00, 0x00, 0x00, 0x00, 0xff, 0xe0, 0x00, 0x00, + 0x00, 0x01, 0xe0, 0xf0, 0x00, 0x00, 0x00, 0x01, 0xc0, 0x78, 0x00, 0x00, 0x00, 0x03, 0x80, 0x38, 0x00, 0x00, 0x00, 0x01, 0x00, 0x10, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x06, 0x00, 0x00, 0x00, 0x00, 0x00, 0x06, 0x00, 0x00, 0x00, 0x00, 0x00, 0x04, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 +}; + +// 'checkmark', 44x44px +const unsigned char epd_bitmap_checkmark[] PROGMEM = { + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x01, 0x80, 0x00, 0x00, 0x00, 0x00, 0x03, 0x80, 0x00, 0x00, 0x00, 0x00, 0x07, 0x80, 0x00, 0x00, 0x00, 0x00, 0x0f, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x1e, 0x00, 0x00, 0x00, 0x00, 0x00, 0x3c, 0x00, 0x00, 0x00, 0x00, 0x00, 0x78, 0x00, 0x00, 0x00, 0x0e, 0x00, 0xf0, 0x00, 0x00, + 0x00, 0x0f, 0x01, 0xe0, 0x00, 0x00, 0x00, 0x0f, 0x83, 0xc0, 0x00, 0x00, 0x00, 0x07, 0xc7, 0x80, 0x00, 0x00, 0x00, 0x03, 0xef, 0x00, 0x00, 0x00, + 0x00, 0x01, 0xfe, 0x00, 0x00, 0x00, 0x00, 0x00, 0xfc, 0x00, 0x00, 0x00, 0x00, 0x00, 0x78, 0x00, 0x00, 0x00, 0x00, 0x00, 0x10, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 +}; + +void IRAM_ATTR handleButtonPress() { + unsigned long currentTime = millis(); // Get current time + + // Debounce: If the current press is too close to the last one, ignore it + if (currentTime - lastPressTime > 50) { + pressCount++; // Count the button press + + // Check if this is the second press within the double-press interval + if (pressCount == 2 && (currentTime - lastPressTime <= doublePressInterval)) { + doublePressDetected = true; // Double press detected + pressCount = 0; // Reset counter + } + + lastPressTime = currentTime; // Update the time of the last press + } +} + +// Function to switch the boot partition to OTA1 +void setBootPartitionToOTA0() { + const esp_partition_t *ota0_partition = esp_partition_find_first(ESP_PARTITION_TYPE_APP, ESP_PARTITION_SUBTYPE_APP_OTA_0, NULL); + + if (ota0_partition) { + // Set OTA1 as new boot partition + esp_ota_set_boot_partition(ota0_partition); + Serial.println("Boot partition changed to OTA0. Restarting..."); + + // Restart to boot from the new partition + esp_restart(); + } else { + Serial.println("OTA1 partition not found!"); + } +} + +void setupDisplay() { + displayEnabled = display.begin(SSD1306_SWITCHCAPVCC, 0x3D); + if (displayEnabled) { + display.display(); + delay(100); + display.clearDisplay(); + } +} + +void displayStatusBar(int progress) { + display.clearDisplay(); + display.setCursor(24, 8); + display.println("Sketch wird"); + display.setCursor(22, 22); + display.println("hochgeladen!"); + + display.fillRect(0, SCREEN_HEIGHT - 24, SCREEN_WIDTH - 4, 8, BLACK); // Clear status bar area + display.drawRect(0, SCREEN_HEIGHT - 24, SCREEN_WIDTH - 4, 8, WHITE); // Draw border + int filledWidth = (progress * SCREEN_WIDTH - 4) / 100; // Calculate progress width + display.fillRect(1, SCREEN_HEIGHT - 23, filledWidth - 4, 6, WHITE); // Fill progress bar + + display.setCursor((SCREEN_WIDTH / 2) - 12, SCREEN_HEIGHT - 10); + display.setTextSize(1); + display.setTextColor(WHITE, BLACK); + display.print(progress); + display.println(" %"); + display.display(); +} + +void displayWelcomeScreen() { + display.clearDisplay(); + + // Draw WiFi symbol + display.drawBitmap(0, 12, epd_bitmap_wifi, 44, 44, WHITE); + + // Display SSID text + display.setCursor(40, 13); + display.setTextSize(1); + display.setTextColor(WHITE, BLACK); + display.println("Verbinde dich"); // "Connect" + display.setCursor(60, 27); + display.println("mit:"); // "with" + + // Display SSID + display.setCursor(40, 43); + display.setTextSize(1); // Larger text for SSID + display.print(ssid); + + display.display(); +} + +void displaySuccessScreen() { + display.clearDisplay(); + + // Draw WiFi symbol + display.drawBitmap(0, 12, epd_bitmap_checkmark, 44, 44, WHITE); + + // Display SSID text + display.setCursor(48, 22); + display.setTextSize(1); + display.setTextColor(WHITE, BLACK); + display.println("Erfolgreich"); // "Successfully" + display.setCursor(48, 36); + display.println("hochgeladen!"); // "uploaded!" + + display.display(); +} + +void wipeDisplay() { + display.clearDisplay(); + display.println(""); + display.display(); +} + +void setupWiFi() { + WiFi.macAddress(mac); + char macLastFour[5]; + snprintf(macLastFour, sizeof(macLastFour), "%02X%02X", mac[4], mac[5]); + ssid = "senseBox:" + String(macLastFour); + + // Define the IP address, gateway, and subnet mask + IPAddress local_IP(192, 168, 1, 1); // The new IP address + IPAddress gateway(192, 168, 1, 1); // Gateway address (can be the same as the AP's IP) + IPAddress subnet(255, 255, 255, 0); // Subnet mask + + // Set the IP address, gateway, and subnet mask of the access point + WiFi.softAPConfig(local_IP, gateway, subnet); + + // Start the access point + WiFi.softAP(ssid.c_str()); +} + +void setupOTA() { + // Handle updating process + server.on( + "/sketch", HTTP_POST, + []() { + server.sendHeader("Connection", "close"); + server.send(200, "text/plain", (Update.hasError()) ? "FAIL" : "OK"); + ESP.restart(); + }, + []() { + HTTPUpload &upload = server.upload(); + + if (upload.status == UPLOAD_FILE_START) { + Serial.setDebugOutput(true); + size_t fsize = UPDATE_SIZE_UNKNOWN; + if (server.clientContentLength() > 0) { + fsize = server.clientContentLength(); + } + Serial.printf("Receiving Update: %s, Size: %d\n", upload.filename.c_str(), fsize); + + Serial.printf("Update: %s\n", upload.filename.c_str()); + if (!Update.begin(fsize)) { //start with max available size + Update.printError(Serial); + } + } else if (upload.status == UPLOAD_FILE_WRITE) { + /* flashing firmware to ESP*/ + if (Update.write(upload.buf, upload.currentSize) != upload.currentSize) { + Update.printError(Serial); + } else { + int progress = (Update.progress() * 100) / Update.size(); + displayStatusBar(progress); // Update progress on status bar + } + } else if (upload.status == UPLOAD_FILE_END) { + if (Update.end(true)) { //true to set the size to the current progress + displaySuccessScreen(); + delay(3000); + wipeDisplay(); + } else { + Update.printError(Serial); + } + Serial.setDebugOutput(false); + } + yield(); + } + ); +} + +void setup() { + // Start Serial communication + Serial.begin(115200); + rgb_led_1.begin(); + rgb_led_1.setBrightness(30); + rgb_led_1.setPixelColor(0, rgb_led_1.Color(51, 51, 255)); + rgb_led_1.show(); + + // Configure button pin as input + pinMode(BUTTON_PIN, INPUT_PULLUP); + + // Interrupt for the button + attachInterrupt(digitalPinToInterrupt(BUTTON_PIN), handleButtonPress, FALLING); + +#ifdef DISPLAY_ENABLED + setupDisplay(); +#endif + setupWiFi(); + // Set the ESP32 as an access point + setupOTA(); + server.begin(); +} + +void loop() { + // Handle client requests + server.handleClient(); + +#ifdef DISPLAY_ENABLED + displayWelcomeScreen(); +#endif + + if (doublePressDetected) { + Serial.println("Doppeldruck erkannt!"); // "Double press detected!" + setBootPartitionToOTA0(); +#ifdef DISPLAY_ENABLED + display.setCursor(0, 0); + display.setTextSize(1); + display.setTextColor(WHITE, BLACK); + display.println(""); + display.display(); + delay(50); +#endif + // Restart to boot from the new partition + esp_restart(); + } +} diff --git a/variants/sensebox_mcu_esp32s2/variant.cpp b/variants/sensebox_mcu_esp32s2/variant.cpp index 0c58ef2cbe2..aa1eb3dc7c5 100644 --- a/variants/sensebox_mcu_esp32s2/variant.cpp +++ b/variants/sensebox_mcu_esp32s2/variant.cpp @@ -1,29 +1,10 @@ -/* - * The MIT License (MIT) - * - * Copyright (c) 2021 Ha Thach (tinyusb.org) for Adafruit Industries - * - * Permission is hereby granted, free of charge, to any person obtaining a copy - * of this software and associated documentation files (the "Software"), to deal - * in the Software without restriction, including without limitation the rights - * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell - * copies of the Software, and to permit persons to whom the Software is - * furnished to do so, subject to the following conditions: - * - * The above copyright notice and this permission notice shall be included in - * all copies or substantial portions of the Software. - * - * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR - * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, - * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE - * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER - * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, - * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN - * THE SOFTWARE. - */ - #include "esp32-hal-gpio.h" #include "pins_arduino.h" +#include "esp_partition.h" +#include "esp_system.h" +#include "esp_ota_ops.h" +#include "esp_log.h" +#include extern "C" { @@ -41,12 +22,51 @@ void initVariant(void) { pinMode(PIN_XB1_ENABLE, OUTPUT); digitalWrite(PIN_XB1_ENABLE, LOW); - //enable UART by default - pinMode(PIN_UART_ENABLE, OUTPUT); - digitalWrite(PIN_UART_ENABLE, LOW); + //enable UART only for chip without PSRAM + esp_chip_info_t chip_info; + esp_chip_info(&chip_info); + if (chip_info.revision <= 0) { + pinMode(PIN_UART_ENABLE, OUTPUT); + digitalWrite(PIN_UART_ENABLE, LOW); + } //enable PD-Sensor by default pinMode(PD_ENABLE, OUTPUT); digitalWrite(PD_ENABLE, HIGH); + + // define button pin + const int PIN_BUTTON = 0; + pinMode(PIN_BUTTON, INPUT_PULLUP); + + // keep button pressed + unsigned long pressStartTime = 0; + bool buttonPressed = false; + + // Wait 5 seconds for the button to be pressed + unsigned long startTime = millis(); + + // Check if button is pressed + while (millis() - startTime < 5000) { + if (digitalRead(PIN_BUTTON) == LOW) { + if (!buttonPressed) { + // The button was pressed + buttonPressed = true; + } + } else if (buttonPressed) { + // When the button is pressed and then released, boot into the OTA1 partition + const esp_partition_t *ota1_partition = esp_partition_find_first(ESP_PARTITION_TYPE_APP, ESP_PARTITION_SUBTYPE_APP_OTA_1, NULL); + + if (ota1_partition) { + esp_err_t err = esp_ota_set_boot_partition(ota1_partition); + if (err == ESP_OK) { + esp_restart(); // restart, to boot OTA1 partition + } else { + ESP_LOGE("OTA", "Error setting OTA1 partition: %s", esp_err_to_name(err)); + } + } + // Abort after releasing the button + break; + } + } } } From d4e5c5f969aba994e3e1c1994348bffb9856e95a Mon Sep 17 00:00:00 2001 From: Me No Dev Date: Thu, 3 Jul 2025 12:25:33 +0300 Subject: [PATCH 089/173] IDF release/v5.5 adb3f2a5 (#11543) IDF release/v5.5 adb3f2a5 --- package/package_esp32_index.template.json | 68 +++++++++++------------ 1 file changed, 34 insertions(+), 34 deletions(-) diff --git a/package/package_esp32_index.template.json b/package/package_esp32_index.template.json index 1d753792b43..a02be509094 100644 --- a/package/package_esp32_index.template.json +++ b/package/package_esp32_index.template.json @@ -51,7 +51,7 @@ { "packager": "esp32", "name": "esp32-arduino-libs", - "version": "idf-release_v5.5-cbe9388f-v1" + "version": "idf-release_v5.5-adb3f2a5-v1" }, { "packager": "esp32", @@ -104,63 +104,63 @@ "tools": [ { "name": "esp32-arduino-libs", - "version": "idf-release_v5.5-cbe9388f-v1", + "version": "idf-release_v5.5-adb3f2a5-v1", "systems": [ { "host": "i686-mingw32", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-cbe9388f-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-cbe9388f-v1.zip", - "checksum": "SHA-256:b737ffb86a1b377db12dd610d06936ca8d85d877c872f532a68f6f0a3f666a3f", - "size": "421300036" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-adb3f2a5-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-adb3f2a5-v1.zip", + "checksum": "SHA-256:c7bdda06e7ddae51880fc0e1c76114a8e766a9d682e3645e7794f7126b895a94", + "size": "430508851" }, { "host": "x86_64-mingw32", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-cbe9388f-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-cbe9388f-v1.zip", - "checksum": "SHA-256:b737ffb86a1b377db12dd610d06936ca8d85d877c872f532a68f6f0a3f666a3f", - "size": "421300036" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-adb3f2a5-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-adb3f2a5-v1.zip", + "checksum": "SHA-256:c7bdda06e7ddae51880fc0e1c76114a8e766a9d682e3645e7794f7126b895a94", + "size": "430508851" }, { "host": "arm64-apple-darwin", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-cbe9388f-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-cbe9388f-v1.zip", - "checksum": "SHA-256:b737ffb86a1b377db12dd610d06936ca8d85d877c872f532a68f6f0a3f666a3f", - "size": "421300036" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-adb3f2a5-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-adb3f2a5-v1.zip", + "checksum": "SHA-256:c7bdda06e7ddae51880fc0e1c76114a8e766a9d682e3645e7794f7126b895a94", + "size": "430508851" }, { "host": "x86_64-apple-darwin", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-cbe9388f-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-cbe9388f-v1.zip", - "checksum": "SHA-256:b737ffb86a1b377db12dd610d06936ca8d85d877c872f532a68f6f0a3f666a3f", - "size": "421300036" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-adb3f2a5-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-adb3f2a5-v1.zip", + "checksum": "SHA-256:c7bdda06e7ddae51880fc0e1c76114a8e766a9d682e3645e7794f7126b895a94", + "size": "430508851" }, { "host": "x86_64-pc-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-cbe9388f-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-cbe9388f-v1.zip", - "checksum": "SHA-256:b737ffb86a1b377db12dd610d06936ca8d85d877c872f532a68f6f0a3f666a3f", - "size": "421300036" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-adb3f2a5-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-adb3f2a5-v1.zip", + "checksum": "SHA-256:c7bdda06e7ddae51880fc0e1c76114a8e766a9d682e3645e7794f7126b895a94", + "size": "430508851" }, { "host": "i686-pc-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-cbe9388f-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-cbe9388f-v1.zip", - "checksum": "SHA-256:b737ffb86a1b377db12dd610d06936ca8d85d877c872f532a68f6f0a3f666a3f", - "size": "421300036" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-adb3f2a5-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-adb3f2a5-v1.zip", + "checksum": "SHA-256:c7bdda06e7ddae51880fc0e1c76114a8e766a9d682e3645e7794f7126b895a94", + "size": "430508851" }, { "host": "aarch64-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-cbe9388f-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-cbe9388f-v1.zip", - "checksum": "SHA-256:b737ffb86a1b377db12dd610d06936ca8d85d877c872f532a68f6f0a3f666a3f", - "size": "421300036" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-adb3f2a5-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-adb3f2a5-v1.zip", + "checksum": "SHA-256:c7bdda06e7ddae51880fc0e1c76114a8e766a9d682e3645e7794f7126b895a94", + "size": "430508851" }, { "host": "arm-linux-gnueabihf", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-cbe9388f-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-cbe9388f-v1.zip", - "checksum": "SHA-256:b737ffb86a1b377db12dd610d06936ca8d85d877c872f532a68f6f0a3f666a3f", - "size": "421300036" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-adb3f2a5-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-adb3f2a5-v1.zip", + "checksum": "SHA-256:c7bdda06e7ddae51880fc0e1c76114a8e766a9d682e3645e7794f7126b895a94", + "size": "430508851" } ] }, From c2d23258f197d0f3422c31e75ab848919fc097bb Mon Sep 17 00:00:00 2001 From: Sugar Glider Date: Thu, 3 Jul 2025 06:25:51 -0300 Subject: [PATCH 090/173] feat(matter): enables BLE Matter commissioning with NimBLE (#11537) * feat(matter): enables BLE Matter commissioning with NimBLE * fix(matter): commentary typo and formatting * fix(matter): commentary typo and formatting * fix(matter): removes forcing second network clustter * fix(matter): adds matter source code to CMakeLists.txt * ci(pre-commit): Apply automatic fixes --------- Co-authored-by: pre-commit-ci-lite[bot] <117423508+pre-commit-ci-lite[bot]@users.noreply.github.com> --- CMakeLists.txt | 3 +- .../MatterColorLight/MatterColorLight.ino | 15 +- .../MatterCommissionTest.ino | 13 +- .../MatterComposedLights.ino | 13 +- .../MatterContactSensor.ino | 13 +- .../MatterDimmableLight.ino | 15 +- .../MatterEnhancedColorLight.ino | 15 +- .../examples/MatterEvents/MatterEvents.ino | 9 ++ .../Matter/examples/MatterFan/MatterFan.ino | 15 +- .../MatterHumiditySensor.ino | 13 +- .../examples/MatterMinimum/MatterMinimum.ino | 11 +- .../MatterOccupancySensor.ino | 13 +- .../MatterOnIdentify/MatterOnIdentify.ino | 13 +- .../MatterOnOffLight/MatterOnOffLight.ino | 15 +- .../MatterOnOffPlugin/MatterOnOffPlugin.ino | 15 +- .../MatterPressureSensor.ino | 13 +- .../MatterSmartButon/MatterSmartButon.ino | 15 +- .../MatterTemperatureLight.ino | 15 +- .../MatterTemperatureSensor.ino | 13 +- .../MatterThermostat/MatterThermostat.ino | 13 +- libraries/Matter/src/Matter.cpp | 18 ++- libraries/Matter/src/Matter.h | 2 +- libraries/Matter/src/MatterEndPoint.cpp | 139 ++++++++++++++++++ libraries/Matter/src/MatterEndPoint.h | 117 +++++---------- .../src/MatterEndpoints/MatterColorLight.cpp | 2 +- .../src/MatterEndpoints/MatterColorLight.h | 2 +- .../MatterColorTemperatureLight.cpp | 2 +- .../MatterColorTemperatureLight.h | 2 +- .../MatterEndpoints/MatterContactSensor.cpp | 3 +- .../src/MatterEndpoints/MatterContactSensor.h | 2 +- .../MatterEndpoints/MatterDimmableLight.cpp | 2 +- .../src/MatterEndpoints/MatterDimmableLight.h | 2 +- .../MatterEnhancedColorLight.cpp | 2 +- .../MatterEnhancedColorLight.h | 2 +- .../Matter/src/MatterEndpoints/MatterFan.cpp | 3 +- .../Matter/src/MatterEndpoints/MatterFan.h | 2 +- .../MatterEndpoints/MatterGenericSwitch.cpp | 3 +- .../src/MatterEndpoints/MatterGenericSwitch.h | 2 +- .../MatterEndpoints/MatterHumiditySensor.cpp | 3 +- .../MatterEndpoints/MatterHumiditySensor.h | 2 +- .../MatterEndpoints/MatterOccupancySensor.cpp | 3 +- .../MatterEndpoints/MatterOccupancySensor.h | 2 +- .../src/MatterEndpoints/MatterOnOffLight.cpp | 3 +- .../src/MatterEndpoints/MatterOnOffLight.h | 2 +- .../src/MatterEndpoints/MatterOnOffPlugin.cpp | 3 +- .../src/MatterEndpoints/MatterOnOffPlugin.h | 2 +- .../MatterEndpoints/MatterPressureSensor.cpp | 3 +- .../MatterEndpoints/MatterPressureSensor.h | 2 +- .../MatterTemperatureSensor.cpp | 3 +- .../MatterEndpoints/MatterTemperatureSensor.h | 2 +- .../src/MatterEndpoints/MatterThermostat.cpp | 3 +- .../src/MatterEndpoints/MatterThermostat.h | 2 +- 52 files changed, 447 insertions(+), 155 deletions(-) create mode 100644 libraries/Matter/src/MatterEndPoint.cpp diff --git a/CMakeLists.txt b/CMakeLists.txt index 20cfc2b0955..e71911c1e38 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -184,7 +184,8 @@ set(ARDUINO_LIBRARY_Matter_SRCS libraries/Matter/src/MatterEndpoints/MatterOccupancySensor.cpp libraries/Matter/src/MatterEndpoints/MatterOnOffPlugin.cpp libraries/Matter/src/MatterEndpoints/MatterThermostat.cpp - libraries/Matter/src/Matter.cpp) + libraries/Matter/src/Matter.cpp + libraries/Matter/src/MatterEndPoint.cpp) set(ARDUINO_LIBRARY_PPP_SRCS libraries/PPP/src/PPP.cpp diff --git a/libraries/Matter/examples/MatterColorLight/MatterColorLight.ino b/libraries/Matter/examples/MatterColorLight/MatterColorLight.ino index f3e45887576..e28c96e266c 100644 --- a/libraries/Matter/examples/MatterColorLight/MatterColorLight.ino +++ b/libraries/Matter/examples/MatterColorLight/MatterColorLight.ino @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. @@ -14,16 +14,22 @@ // Matter Manager #include +#if !CONFIG_ENABLE_CHIPOBLE +// if the device can be commissioned using BLE, WiFi is not used - save flash space #include +#endif #include // List of Matter Endpoints for this Node // Color Light Endpoint MatterColorLight ColorLight; +// CONFIG_ENABLE_CHIPOBLE is enabled when BLE is used to commission the Matter Network +#if !CONFIG_ENABLE_CHIPOBLE // WiFi is manually set and started const char *ssid = "your-ssid"; // Change this to your WiFi SSID const char *password = "your-password"; // Change this to your WiFi password +#endif // it will keep last OnOff & HSV Color state stored, using Preferences Preferences matterPref; @@ -81,6 +87,8 @@ void setup() { Serial.begin(115200); +// CONFIG_ENABLE_CHIPOBLE is enabled when BLE is used to commission the Matter Network +#if !CONFIG_ENABLE_CHIPOBLE // We start by connecting to a WiFi network Serial.print("Connecting to "); Serial.println(ssid); @@ -95,6 +103,7 @@ void setup() { Serial.println("IP address: "); Serial.println(WiFi.localIP()); delay(500); +#endif // Initialize Matter EndPoint matterPref.begin("MatterPrefs", false); @@ -121,7 +130,7 @@ void setup() { Matter.begin(); // This may be a restart of a already commissioned Matter accessory if (Matter.isDeviceCommissioned()) { - Serial.println("Matter Node is commissioned and connected to Wi-Fi. Ready for use."); + Serial.println("Matter Node is commissioned and connected to the network. Ready for use."); Serial.printf( "Initial state: %s | RGB Color: (%d,%d,%d) \r\n", ColorLight ? "ON" : "OFF", ColorLight.getColorRGB().r, ColorLight.getColorRGB().g, ColorLight.getColorRGB().b @@ -154,7 +163,7 @@ void loop() { ); // configure the Light based on initial on-off state and its color ColorLight.updateAccessory(); - Serial.println("Matter Node is commissioned and connected to Wi-Fi. Ready for use."); + Serial.println("Matter Node is commissioned and connected to the network. Ready for use."); } // A button is also used to control the light diff --git a/libraries/Matter/examples/MatterCommissionTest/MatterCommissionTest.ino b/libraries/Matter/examples/MatterCommissionTest/MatterCommissionTest.ino index 0e93ed6d155..aa593758548 100644 --- a/libraries/Matter/examples/MatterCommissionTest/MatterCommissionTest.ino +++ b/libraries/Matter/examples/MatterCommissionTest/MatterCommissionTest.ino @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. @@ -14,19 +14,27 @@ // Matter Manager #include +#if !CONFIG_ENABLE_CHIPOBLE +// if the device can be commissioned using BLE, WiFi is not used - save flash space #include +#endif // List of Matter Endpoints for this Node // On/Off Light Endpoint MatterOnOffLight OnOffLight; +// CONFIG_ENABLE_CHIPOBLE is enabled when BLE is used to commission the Matter Network +#if !CONFIG_ENABLE_CHIPOBLE // WiFi is manually set and started const char *ssid = "your-ssid"; // Change this to your WiFi SSID const char *password = "your-password"; // Change this to your WiFi password +#endif void setup() { Serial.begin(115200); +// CONFIG_ENABLE_CHIPOBLE is enabled when BLE is used to commission the Matter Network +#if !CONFIG_ENABLE_CHIPOBLE // We start by connecting to a WiFi network Serial.print("Connecting to "); Serial.println(ssid); @@ -41,6 +49,7 @@ void setup() { Serial.println("IP address: "); Serial.println(WiFi.localIP()); delay(500); +#endif // Initialize at least one Matter EndPoint OnOffLight.begin(); @@ -64,7 +73,7 @@ void loop() { Serial.println("Matter Fabric not commissioned yet. Waiting for commissioning."); } } - Serial.println("Matter Node is commissioned and connected to Wi-Fi."); + Serial.println("Matter Node is commissioned and connected to the network."); Serial.println("====> Decommissioning in 30 seconds. <===="); delay(30000); Matter.decommission(); diff --git a/libraries/Matter/examples/MatterComposedLights/MatterComposedLights.ino b/libraries/Matter/examples/MatterComposedLights/MatterComposedLights.ino index b98cc8e19c9..48a6db8bedd 100644 --- a/libraries/Matter/examples/MatterComposedLights/MatterComposedLights.ino +++ b/libraries/Matter/examples/MatterComposedLights/MatterComposedLights.ino @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. @@ -14,7 +14,10 @@ // Matter Manager #include +#if !CONFIG_ENABLE_CHIPOBLE +// if the device can be commissioned using BLE, WiFi is not used - save flash space #include +#endif // List of Matter Endpoints for this Node // There will be 3 On/Off Light Endpoints in the same Node @@ -22,9 +25,12 @@ MatterOnOffLight Light1; MatterDimmableLight Light2; MatterColorLight Light3; +// CONFIG_ENABLE_CHIPOBLE is enabled when BLE is used to commission the Matter Network +#if !CONFIG_ENABLE_CHIPOBLE // WiFi is manually set and started const char *ssid = "your-ssid"; // Change this to your WiFi SSID const char *password = "your-password"; // Change this to your WiFi password +#endif // set your board USER BUTTON pin here - USED to decommission the Matter Node const uint8_t buttonPin = BOOT_PIN; // Set your pin here. Using BOOT Button. @@ -56,6 +62,8 @@ void setup() { Serial.begin(115200); +// CONFIG_ENABLE_CHIPOBLE is enabled when BLE is used to commission the Matter Network +#if !CONFIG_ENABLE_CHIPOBLE // We start by connecting to a WiFi network Serial.print("Connecting to "); Serial.println(ssid); @@ -71,6 +79,7 @@ void setup() { Serial.println("IP address: "); Serial.println(WiFi.localIP()); delay(500); +#endif // Initialize all 3 Matter EndPoints Light1.begin(); @@ -103,7 +112,7 @@ void loop() { Serial.println("Matter Node not commissioned yet. Waiting for commissioning."); } } - Serial.println("Matter Node is commissioned and connected to Wi-Fi. Ready for use."); + Serial.println("Matter Node is commissioned and connected to the network. Ready for use."); } //displays the Light state every 5 seconds diff --git a/libraries/Matter/examples/MatterContactSensor/MatterContactSensor.ino b/libraries/Matter/examples/MatterContactSensor/MatterContactSensor.ino index e4c41460d3a..6472012a2ff 100644 --- a/libraries/Matter/examples/MatterContactSensor/MatterContactSensor.ino +++ b/libraries/Matter/examples/MatterContactSensor/MatterContactSensor.ino @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. @@ -30,15 +30,21 @@ // Matter Manager #include +#if !CONFIG_ENABLE_CHIPOBLE +// if the device can be commissioned using BLE, WiFi is not used - save flash space #include +#endif // List of Matter Endpoints for this Node // Matter Contact Sensor Endpoint MatterContactSensor ContactSensor; +// CONFIG_ENABLE_CHIPOBLE is enabled when BLE is used to commission the Matter Network +#if !CONFIG_ENABLE_CHIPOBLE // WiFi is manually set and started const char *ssid = "your-ssid"; // Change this to your WiFi SSID const char *password = "your-password"; // Change this to your WiFi password +#endif // LED will be used to indicate the Contact Sensor state // set your board RGB LED pin here @@ -67,6 +73,8 @@ void setup() { Serial.begin(115200); +// CONFIG_ENABLE_CHIPOBLE is enabled when BLE is used to commission the Matter Network +#if !CONFIG_ENABLE_CHIPOBLE // Manually connect to WiFi WiFi.begin(ssid, password); // Wait for connection @@ -75,6 +83,7 @@ void setup() { Serial.print("."); } Serial.println(); +#endif // set initial contact sensor state as false (default) ContactSensor.begin(); @@ -99,7 +108,7 @@ void setup() { Serial.println("Matter Node not commissioned yet. Waiting for commissioning."); } } - Serial.println("Matter Node is commissioned and connected to Wi-Fi. Ready for use."); + Serial.println("Matter Node is commissioned and connected to the network. Ready for use."); } } diff --git a/libraries/Matter/examples/MatterDimmableLight/MatterDimmableLight.ino b/libraries/Matter/examples/MatterDimmableLight/MatterDimmableLight.ino index 79751905c20..2b853b25677 100644 --- a/libraries/Matter/examples/MatterDimmableLight/MatterDimmableLight.ino +++ b/libraries/Matter/examples/MatterDimmableLight/MatterDimmableLight.ino @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. @@ -14,16 +14,22 @@ // Matter Manager #include +#if !CONFIG_ENABLE_CHIPOBLE +// if the device can be commissioned using BLE, WiFi is not used - save flash space #include +#endif #include // List of Matter Endpoints for this Node // Dimmable Light Endpoint MatterDimmableLight DimmableLight; +// CONFIG_ENABLE_CHIPOBLE is enabled when BLE is used to commission the Matter Network +#if !CONFIG_ENABLE_CHIPOBLE // WiFi is manually set and started const char *ssid = "your-ssid"; // Change this to your WiFi SSID const char *password = "your-password"; // Change this to your WiFi password +#endif // it will keep last OnOff & Brightness state stored, using Preferences Preferences matterPref; @@ -77,6 +83,8 @@ void setup() { Serial.begin(115200); +// CONFIG_ENABLE_CHIPOBLE is enabled when BLE is used to commission the Matter Network +#if !CONFIG_ENABLE_CHIPOBLE // We start by connecting to a WiFi network Serial.print("Connecting to "); Serial.println(ssid); @@ -91,6 +99,7 @@ void setup() { Serial.println("IP address: "); Serial.println(WiFi.localIP()); delay(500); +#endif // Initialize Matter EndPoint matterPref.begin("MatterPrefs", false); @@ -116,7 +125,7 @@ void setup() { Matter.begin(); // This may be a restart of a already commissioned Matter accessory if (Matter.isDeviceCommissioned()) { - Serial.println("Matter Node is commissioned and connected to Wi-Fi. Ready for use."); + Serial.println("Matter Node is commissioned and connected to the network. Ready for use."); Serial.printf("Initial state: %s | brightness: %d\r\n", DimmableLight ? "ON" : "OFF", DimmableLight.getBrightness()); // configure the Light based on initial on-off state and brightness DimmableLight.updateAccessory(); @@ -143,7 +152,7 @@ void loop() { Serial.printf("Initial state: %s | brightness: %d\r\n", DimmableLight ? "ON" : "OFF", DimmableLight.getBrightness()); // configure the Light based on initial on-off state and brightness DimmableLight.updateAccessory(); - Serial.println("Matter Node is commissioned and connected to Wi-Fi. Ready for use."); + Serial.println("Matter Node is commissioned and connected to the network. Ready for use."); } // A button is also used to control the light diff --git a/libraries/Matter/examples/MatterEnhancedColorLight/MatterEnhancedColorLight.ino b/libraries/Matter/examples/MatterEnhancedColorLight/MatterEnhancedColorLight.ino index 8e12581fdf2..ac22ae768c5 100644 --- a/libraries/Matter/examples/MatterEnhancedColorLight/MatterEnhancedColorLight.ino +++ b/libraries/Matter/examples/MatterEnhancedColorLight/MatterEnhancedColorLight.ino @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. @@ -14,16 +14,22 @@ // Matter Manager #include +#if !CONFIG_ENABLE_CHIPOBLE +// if the device can be commissioned using BLE, WiFi is not used - save flash space #include +#endif #include // List of Matter Endpoints for this Node // Color Light Endpoint MatterEnhancedColorLight EnhancedColorLight; +// CONFIG_ENABLE_CHIPOBLE is enabled when BLE is used to commission the Matter Network +#if !CONFIG_ENABLE_CHIPOBLE // WiFi is manually set and started const char *ssid = "your-ssid"; // Change this to your WiFi SSID const char *password = "your-password"; // Change this to your WiFi password +#endif // It will use HSV color to control all Matter Attribute Changes HsvColor_t currentHSVColor = {0, 0, 0}; @@ -85,6 +91,8 @@ void setup() { Serial.begin(115200); +// CONFIG_ENABLE_CHIPOBLE is enabled when BLE is used to commission the Matter Network +#if !CONFIG_ENABLE_CHIPOBLE // We start by connecting to a WiFi network Serial.print("Connecting to "); Serial.println(ssid); @@ -99,6 +107,7 @@ void setup() { Serial.println("IP address: "); Serial.println(WiFi.localIP()); delay(500); +#endif // Initialize Matter EndPoint matterPref.begin("MatterPrefs", false); @@ -143,7 +152,7 @@ void setup() { Matter.begin(); // This may be a restart of a already commissioned Matter accessory if (Matter.isDeviceCommissioned()) { - Serial.println("Matter Node is commissioned and connected to Wi-Fi. Ready for use."); + Serial.println("Matter Node is commissioned and connected to the network. Ready for use."); Serial.printf( "Initial state: %s | RGB Color: (%d,%d,%d) \r\n", EnhancedColorLight ? "ON" : "OFF", EnhancedColorLight.getColorRGB().r, EnhancedColorLight.getColorRGB().g, EnhancedColorLight.getColorRGB().b @@ -176,7 +185,7 @@ void loop() { ); // configure the Light based on initial on-off state and its color EnhancedColorLight.updateAccessory(); - Serial.println("Matter Node is commissioned and connected to Wi-Fi. Ready for use."); + Serial.println("Matter Node is commissioned and connected to the network. Ready for use."); } // A button is also used to control the light diff --git a/libraries/Matter/examples/MatterEvents/MatterEvents.ino b/libraries/Matter/examples/MatterEvents/MatterEvents.ino index dac599bf9fa..f33b13cf7fb 100644 --- a/libraries/Matter/examples/MatterEvents/MatterEvents.ino +++ b/libraries/Matter/examples/MatterEvents/MatterEvents.ino @@ -14,11 +14,17 @@ // Matter Manager #include +#if !CONFIG_ENABLE_CHIPOBLE +// if the device can be commissioned using BLE, WiFi is not used - save flash space #include +#endif +// CONFIG_ENABLE_CHIPOBLE is enabled when BLE is used to commission the Matter Network +#if !CONFIG_ENABLE_CHIPOBLE // WiFi is manually set and started const char *ssid = "your-ssid"; // Change this to your WiFi SSID const char *password = "your-password"; // Change this to your WiFi password +#endif // List of Matter Endpoints for this Node // On/Off Light Endpoint @@ -119,6 +125,8 @@ void setup() { delay(10); // Wait for Serial to initialize } +// CONFIG_ENABLE_CHIPOBLE is enabled when BLE is used to commission the Matter Network +#if !CONFIG_ENABLE_CHIPOBLE // We start by connecting to a WiFi network Serial.print("Connecting to "); Serial.println(ssid); @@ -134,6 +142,7 @@ void setup() { Serial.println("IP address: "); Serial.println(WiFi.localIP()); delay(500); +#endif // Initialize at least one Matter EndPoint OnOffLight.begin(); diff --git a/libraries/Matter/examples/MatterFan/MatterFan.ino b/libraries/Matter/examples/MatterFan/MatterFan.ino index 705aa4853da..ac620167a40 100644 --- a/libraries/Matter/examples/MatterFan/MatterFan.ino +++ b/libraries/Matter/examples/MatterFan/MatterFan.ino @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. @@ -14,15 +14,21 @@ // Matter Manager #include +#if !CONFIG_ENABLE_CHIPOBLE +// if the device can be commissioned using BLE, WiFi is not used - save flash space #include +#endif // List of Matter Endpoints for this Node // Fan Endpoint - On/Off control + Speed Percent Control + Fan Modes MatterFan Fan; +// CONFIG_ENABLE_CHIPOBLE is enabled when BLE is used to commission the Matter Network +#if !CONFIG_ENABLE_CHIPOBLE // WiFi is manually set and started const char *ssid = "your-ssid"; // Change this to your WiFi SSID const char *password = "your-password"; // Change this to your WiFi password +#endif // set your board USER BUTTON pin here - used for toggling On/Off and decommission the Matter Node const uint8_t buttonPin = BOOT_PIN; // Set your pin here. Using BOOT Button. @@ -76,6 +82,8 @@ void setup() { Serial.begin(115200); +// CONFIG_ENABLE_CHIPOBLE is enabled when BLE is used to commission the Matter Network +#if !CONFIG_ENABLE_CHIPOBLE // We start by connecting to a WiFi network Serial.print("Connecting to "); Serial.println(ssid); @@ -90,6 +98,7 @@ void setup() { Serial.println("IP address: "); Serial.println(WiFi.localIP()); delay(500); +#endif // On Boot or Reset, Fan is set at 0% speed, OFF, changing between OFF, ON, SMART and HIGH Fan.begin(0, MatterFan::FAN_MODE_OFF, MatterFan::FAN_MODE_SEQ_OFF_HIGH); @@ -141,7 +150,7 @@ void setup() { Matter.begin(); // This may be a restart of a already commissioned Matter accessory if (Matter.isDeviceCommissioned()) { - Serial.println("Matter Node is commissioned and connected to Wi-Fi. Ready for use."); + Serial.println("Matter Node is commissioned and connected to the network. Ready for use."); } } @@ -162,7 +171,7 @@ void loop() { Serial.println("Matter Node not commissioned yet. Waiting for commissioning."); } } - Serial.println("Matter Node is commissioned and connected to Wi-Fi. Ready for use."); + Serial.println("Matter Node is commissioned and connected to the network. Ready for use."); } // A builtin button is used to trigger and send a command to the Matter Controller diff --git a/libraries/Matter/examples/MatterHumiditySensor/MatterHumiditySensor.ino b/libraries/Matter/examples/MatterHumiditySensor/MatterHumiditySensor.ino index 1c7889db849..3149cf1dfbe 100644 --- a/libraries/Matter/examples/MatterHumiditySensor/MatterHumiditySensor.ino +++ b/libraries/Matter/examples/MatterHumiditySensor/MatterHumiditySensor.ino @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. @@ -21,15 +21,21 @@ // Matter Manager #include +#if !CONFIG_ENABLE_CHIPOBLE +// if the device can be commissioned using BLE, WiFi is not used - save flash space #include +#endif // List of Matter Endpoints for this Node // Matter Humidity Sensor Endpoint MatterHumiditySensor SimulatedHumiditySensor; +// CONFIG_ENABLE_CHIPOBLE is enabled when BLE is used to commission the Matter Network +#if !CONFIG_ENABLE_CHIPOBLE // WiFi is manually set and started const char *ssid = "your-ssid"; // Change this to your WiFi SSID const char *password = "your-password"; // Change this to your WiFi password +#endif // set your board USER BUTTON pin here - decommissioning button const uint8_t buttonPin = BOOT_PIN; // Set your pin here. Using BOOT Button. @@ -60,6 +66,8 @@ void setup() { Serial.begin(115200); +// CONFIG_ENABLE_CHIPOBLE is enabled when BLE is used to commission the Matter Network +#if !CONFIG_ENABLE_CHIPOBLE // Manually connect to WiFi WiFi.begin(ssid, password); // Wait for connection @@ -68,6 +76,7 @@ void setup() { Serial.print("."); } Serial.println(); +#endif // set initial humidity sensor measurement // Simulated Sensor - it shall initially print 95% and then move to the 10% to 30% humidity range @@ -92,7 +101,7 @@ void setup() { Serial.println("Matter Node not commissioned yet. Waiting for commissioning."); } } - Serial.println("Matter Node is commissioned and connected to Wi-Fi. Ready for use."); + Serial.println("Matter Node is commissioned and connected to the network. Ready for use."); } } diff --git a/libraries/Matter/examples/MatterMinimum/MatterMinimum.ino b/libraries/Matter/examples/MatterMinimum/MatterMinimum.ino index db591ee2226..31599ad10cb 100644 --- a/libraries/Matter/examples/MatterMinimum/MatterMinimum.ino +++ b/libraries/Matter/examples/MatterMinimum/MatterMinimum.ino @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. @@ -22,15 +22,21 @@ // Matter Manager #include +#if !CONFIG_ENABLE_CHIPOBLE +// if the device can be commissioned using BLE, WiFi is not used - save flash space #include +#endif // List of Matter Endpoints for this Node // Single On/Off Light Endpoint - at least one per node MatterOnOffLight OnOffLight; +// CONFIG_ENABLE_CHIPOBLE is enabled when BLE is used to commission the Matter Network +#if !CONFIG_ENABLE_CHIPOBLE // WiFi is manually set and started const char *ssid = "your-ssid"; // Change this to your WiFi SSID const char *password = "your-password"; // Change this to your WiFi password +#endif // Light GPIO that can be controlled by Matter APP #ifdef LED_BUILTIN @@ -62,6 +68,8 @@ void setup() { // Initialize the LED GPIO pinMode(ledPin, OUTPUT); +// CONFIG_ENABLE_CHIPOBLE is enabled when BLE is used to commission the Matter Network +#if !CONFIG_ENABLE_CHIPOBLE // Manually connect to WiFi WiFi.begin(ssid, password); // Wait for connection @@ -73,6 +81,7 @@ void setup() { delay(500); } Serial.println(); +#endif // Initialize at least one Matter EndPoint OnOffLight.begin(); diff --git a/libraries/Matter/examples/MatterOccupancySensor/MatterOccupancySensor.ino b/libraries/Matter/examples/MatterOccupancySensor/MatterOccupancySensor.ino index 333f178e9de..ecab016b473 100644 --- a/libraries/Matter/examples/MatterOccupancySensor/MatterOccupancySensor.ino +++ b/libraries/Matter/examples/MatterOccupancySensor/MatterOccupancySensor.ino @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. @@ -28,15 +28,21 @@ // Matter Manager #include +#if !CONFIG_ENABLE_CHIPOBLE +// if the device can be commissioned using BLE, WiFi is not used - save flash space #include +#endif // List of Matter Endpoints for this Node // Matter Occupancy Sensor Endpoint MatterOccupancySensor OccupancySensor; +// CONFIG_ENABLE_CHIPOBLE is enabled when BLE is used to commission the Matter Network +#if !CONFIG_ENABLE_CHIPOBLE // WiFi is manually set and started const char *ssid = "your-ssid"; // Change this to your WiFi SSID const char *password = "your-password"; // Change this to your WiFi password +#endif // set your board USER BUTTON pin here - decommissioning only const uint8_t buttonPin = BOOT_PIN; // Set your pin here. Using BOOT Button. @@ -52,6 +58,8 @@ void setup() { Serial.begin(115200); +// CONFIG_ENABLE_CHIPOBLE is enabled when BLE is used to commission the Matter Network +#if !CONFIG_ENABLE_CHIPOBLE // Manually connect to WiFi WiFi.begin(ssid, password); // Wait for connection @@ -60,6 +68,7 @@ void setup() { Serial.print("."); } Serial.println(); +#endif // set initial occupancy sensor state as false and connected to a PIR sensor type (default) OccupancySensor.begin(); @@ -83,7 +92,7 @@ void setup() { Serial.println("Matter Node not commissioned yet. Waiting for commissioning."); } } - Serial.println("Matter Node is commissioned and connected to Wi-Fi. Ready for use."); + Serial.println("Matter Node is commissioned and connected to the network. Ready for use."); } } diff --git a/libraries/Matter/examples/MatterOnIdentify/MatterOnIdentify.ino b/libraries/Matter/examples/MatterOnIdentify/MatterOnIdentify.ino index b2e77900e95..ec7129ecad9 100644 --- a/libraries/Matter/examples/MatterOnIdentify/MatterOnIdentify.ino +++ b/libraries/Matter/examples/MatterOnIdentify/MatterOnIdentify.ino @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. @@ -26,15 +26,21 @@ // Matter Manager #include +#if !CONFIG_ENABLE_CHIPOBLE +// if the device can be commissioned using BLE, WiFi is not used - save flash space #include +#endif // List of Matter Endpoints for this Node // Single On/Off Light Endpoint - at least one per node MatterOnOffLight OnOffLight; +// CONFIG_ENABLE_CHIPOBLE is enabled when BLE is used to commission the Matter Network +#if !CONFIG_ENABLE_CHIPOBLE // WiFi is manually set and started const char *ssid = "your-ssid"; // Change this to your WiFi SSID const char *password = "your-password"; // Change this to your WiFi password +#endif // Light GPIO that can be controlled by Matter APP #ifdef LED_BUILTIN @@ -88,6 +94,8 @@ void setup() { Serial.begin(115200); +// CONFIG_ENABLE_CHIPOBLE is enabled when BLE is used to commission the Matter Network +#if !CONFIG_ENABLE_CHIPOBLE // Manually connect to WiFi WiFi.begin(ssid, password); // Wait for connection @@ -96,6 +104,7 @@ void setup() { Serial.print("."); } Serial.println(); +#endif // Initialize at least one Matter EndPoint OnOffLight.begin(); @@ -125,7 +134,7 @@ void setup() { Serial.println("Matter Node not commissioned yet. Waiting for commissioning."); } } - Serial.println("Matter Node is commissioned and connected to Wi-Fi. Ready for use."); + Serial.println("Matter Node is commissioned and connected to the network. Ready for use."); } } diff --git a/libraries/Matter/examples/MatterOnOffLight/MatterOnOffLight.ino b/libraries/Matter/examples/MatterOnOffLight/MatterOnOffLight.ino index 5faa0a385b0..3310cb8c4e9 100644 --- a/libraries/Matter/examples/MatterOnOffLight/MatterOnOffLight.ino +++ b/libraries/Matter/examples/MatterOnOffLight/MatterOnOffLight.ino @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. @@ -14,12 +14,18 @@ // Matter Manager #include +#if !CONFIG_ENABLE_CHIPOBLE +// if the device can be commissioned using BLE, WiFi is not used - save flash space #include +#endif #include +// CONFIG_ENABLE_CHIPOBLE is enabled when BLE is used to commission the Matter Network +#if !CONFIG_ENABLE_CHIPOBLE // WiFi is manually set and started const char *ssid = "your-ssid"; // Change this to your WiFi SSID const char *password = "your-password"; // Change this to your WiFi password +#endif // List of Matter Endpoints for this Node // On/Off Light Endpoint @@ -68,6 +74,8 @@ void setup() { Serial.begin(115200); +// CONFIG_ENABLE_CHIPOBLE is enabled when BLE is used to commission the Matter Network +#if !CONFIG_ENABLE_CHIPOBLE // We start by connecting to a WiFi network Serial.print("Connecting to "); Serial.println(ssid); @@ -82,6 +90,7 @@ void setup() { Serial.println("IP address: "); Serial.println(WiFi.localIP()); delay(500); +#endif // Initialize Matter EndPoint matterPref.begin("MatterPrefs", false); @@ -93,7 +102,7 @@ void setup() { Matter.begin(); // This may be a restart of a already commissioned Matter accessory if (Matter.isDeviceCommissioned()) { - Serial.println("Matter Node is commissioned and connected to Wi-Fi. Ready for use."); + Serial.println("Matter Node is commissioned and connected to the network. Ready for use."); Serial.printf("Initial state: %s\r\n", OnOffLight.getOnOff() ? "ON" : "OFF"); OnOffLight.updateAccessory(); // configure the Light based on initial state } @@ -118,7 +127,7 @@ void loop() { } Serial.printf("Initial state: %s\r\n", OnOffLight.getOnOff() ? "ON" : "OFF"); OnOffLight.updateAccessory(); // configure the Light based on initial state - Serial.println("Matter Node is commissioned and connected to Wi-Fi. Ready for use."); + Serial.println("Matter Node is commissioned and connected to the network. Ready for use."); } // A button is also used to control the light diff --git a/libraries/Matter/examples/MatterOnOffPlugin/MatterOnOffPlugin.ino b/libraries/Matter/examples/MatterOnOffPlugin/MatterOnOffPlugin.ino index d14e2189ec1..372874ddc9a 100644 --- a/libraries/Matter/examples/MatterOnOffPlugin/MatterOnOffPlugin.ino +++ b/libraries/Matter/examples/MatterOnOffPlugin/MatterOnOffPlugin.ino @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. @@ -14,16 +14,22 @@ // Matter Manager #include +#if !CONFIG_ENABLE_CHIPOBLE +// if the device can be commissioned using BLE, WiFi is not used - save flash space #include +#endif #include // List of Matter Endpoints for this Node // On/Off Plugin Endpoint MatterOnOffPlugin OnOffPlugin; +// CONFIG_ENABLE_CHIPOBLE is enabled when BLE is used to commission the Matter Network +#if !CONFIG_ENABLE_CHIPOBLE // WiFi is manually set and started const char *ssid = "your-ssid"; // Change this to your WiFi SSID const char *password = "your-password"; // Change this to your WiFi password +#endif // it will keep last OnOff state stored, using Preferences Preferences matterPref; @@ -67,6 +73,8 @@ void setup() { Serial.begin(115200); +// CONFIG_ENABLE_CHIPOBLE is enabled when BLE is used to commission the Matter Network +#if !CONFIG_ENABLE_CHIPOBLE // We start by connecting to a WiFi network Serial.print("Connecting to "); Serial.println(ssid); @@ -80,6 +88,7 @@ void setup() { Serial.println("IP address: "); Serial.println(WiFi.localIP()); delay(500); +#endif // Initialize Matter EndPoint matterPref.begin("MatterPrefs", false); @@ -91,7 +100,7 @@ void setup() { Matter.begin(); // This may be a restart of a already commissioned Matter accessory if (Matter.isDeviceCommissioned()) { - Serial.println("Matter Node is commissioned and connected to Wi-Fi. Ready for use."); + Serial.println("Matter Node is commissioned and connected to the network. Ready for use."); Serial.printf("Initial state: %s\r\n", OnOffPlugin.getOnOff() ? "ON" : "OFF"); OnOffPlugin.updateAccessory(); // configure the Plugin based on initial state } @@ -116,7 +125,7 @@ void loop() { } Serial.printf("Initial state: %s\r\n", OnOffPlugin.getOnOff() ? "ON" : "OFF"); OnOffPlugin.updateAccessory(); // configure the Plugin based on initial state - Serial.println("Matter Node is commissioned and connected to Wi-Fi. Ready for use."); + Serial.println("Matter Node is commissioned and connected to the network. Ready for use."); } // Check if the button has been pressed diff --git a/libraries/Matter/examples/MatterPressureSensor/MatterPressureSensor.ino b/libraries/Matter/examples/MatterPressureSensor/MatterPressureSensor.ino index db035e951c9..0a097ec979b 100644 --- a/libraries/Matter/examples/MatterPressureSensor/MatterPressureSensor.ino +++ b/libraries/Matter/examples/MatterPressureSensor/MatterPressureSensor.ino @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. @@ -21,15 +21,21 @@ // Matter Manager #include +#if !CONFIG_ENABLE_CHIPOBLE +// if the device can be commissioned using BLE, WiFi is not used - save flash space #include +#endif // List of Matter Endpoints for this Node // Matter Pressure Sensor Endpoint MatterPressureSensor SimulatedPressureSensor; +// CONFIG_ENABLE_CHIPOBLE is enabled when BLE is used to commission the Matter Network +#if !CONFIG_ENABLE_CHIPOBLE // WiFi is manually set and started const char *ssid = "your-ssid"; // Change this to your WiFi SSID const char *password = "your-password"; // Change this to your WiFi password +#endif // set your board USER BUTTON pin here - decommissioning button const uint8_t buttonPin = BOOT_PIN; // Set your pin here. Using BOOT Button. @@ -60,6 +66,8 @@ void setup() { Serial.begin(115200); +// CONFIG_ENABLE_CHIPOBLE is enabled when BLE is used to commission the Matter Network +#if !CONFIG_ENABLE_CHIPOBLE // Manually connect to WiFi WiFi.begin(ssid, password); // Wait for connection @@ -68,6 +76,7 @@ void setup() { Serial.print("."); } Serial.println(); +#endif // set initial pressure sensor measurement // Simulated Sensor - it shall initially print 900hPa and then move to the 950 to 1100 hPa as pressure range @@ -92,7 +101,7 @@ void setup() { Serial.println("Matter Node not commissioned yet. Waiting for commissioning."); } } - Serial.println("Matter Node is commissioned and connected to Wi-Fi. Ready for use."); + Serial.println("Matter Node is commissioned and connected to the network. Ready for use."); } } diff --git a/libraries/Matter/examples/MatterSmartButon/MatterSmartButon.ino b/libraries/Matter/examples/MatterSmartButon/MatterSmartButon.ino index f8da970595d..29caf00004c 100644 --- a/libraries/Matter/examples/MatterSmartButon/MatterSmartButon.ino +++ b/libraries/Matter/examples/MatterSmartButon/MatterSmartButon.ino @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. @@ -14,15 +14,21 @@ // Matter Manager #include +#if !CONFIG_ENABLE_CHIPOBLE +// if the device can be commissioned using BLE, WiFi is not used - save flash space #include +#endif // List of Matter Endpoints for this Node // Generic Switch Endpoint - works as a smart button with a single click MatterGenericSwitch SmartButton; +// CONFIG_ENABLE_CHIPOBLE is enabled when BLE is used to commission the Matter Network +#if !CONFIG_ENABLE_CHIPOBLE // WiFi is manually set and started const char *ssid = "your-ssid"; // Change this to your WiFi SSID const char *password = "your-password"; // Change this to your WiFi password +#endif // set your board USER BUTTON pin here const uint8_t buttonPin = BOOT_PIN; // Set your pin here. Using BOOT Button. @@ -43,6 +49,8 @@ void setup() { Serial.print("Connecting to "); Serial.println(ssid); +// CONFIG_ENABLE_CHIPOBLE is enabled when BLE is used to commission the Matter Network +#if !CONFIG_ENABLE_CHIPOBLE // Manually connect to WiFi WiFi.begin(ssid, password); // Wait for connection @@ -54,6 +62,7 @@ void setup() { Serial.println("IP address: "); Serial.println(WiFi.localIP()); delay(500); +#endif // Initialize the Matter EndPoint SmartButton.begin(); @@ -62,7 +71,7 @@ void setup() { Matter.begin(); // This may be a restart of a already commissioned Matter accessory if (Matter.isDeviceCommissioned()) { - Serial.println("Matter Node is commissioned and connected to Wi-Fi. Ready for use."); + Serial.println("Matter Node is commissioned and connected to the network. Ready for use."); } } @@ -83,7 +92,7 @@ void loop() { Serial.println("Matter Node not commissioned yet. Waiting for commissioning."); } } - Serial.println("Matter Node is commissioned and connected to Wi-Fi. Ready for use."); + Serial.println("Matter Node is commissioned and connected to the network. Ready for use."); } // A builtin button is used to trigger a command to the Matter Controller diff --git a/libraries/Matter/examples/MatterTemperatureLight/MatterTemperatureLight.ino b/libraries/Matter/examples/MatterTemperatureLight/MatterTemperatureLight.ino index d46427591ab..b7fcc11d873 100644 --- a/libraries/Matter/examples/MatterTemperatureLight/MatterTemperatureLight.ino +++ b/libraries/Matter/examples/MatterTemperatureLight/MatterTemperatureLight.ino @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. @@ -14,16 +14,22 @@ // Matter Manager #include +#if !CONFIG_ENABLE_CHIPOBLE +// if the device can be commissioned using BLE, WiFi is not used - save flash space #include +#endif #include // List of Matter Endpoints for this Node // Color Temperature CW/WW Light Endpoint MatterColorTemperatureLight CW_WW_Light; +// CONFIG_ENABLE_CHIPOBLE is enabled when BLE is used to commission the Matter Network +#if !CONFIG_ENABLE_CHIPOBLE // WiFi is manually set and started const char *ssid = "your-ssid"; // Change this to your WiFi SSID const char *password = "your-password"; // Change this to your WiFi password +#endif // it will keep last OnOff & Brightness state stored, using Preferences Preferences matterPref; @@ -88,6 +94,8 @@ void setup() { Serial.begin(115200); +// CONFIG_ENABLE_CHIPOBLE is enabled when BLE is used to commission the Matter Network +#if !CONFIG_ENABLE_CHIPOBLE // We start by connecting to a WiFi network Serial.print("Connecting to "); Serial.println(ssid); @@ -102,6 +110,7 @@ void setup() { Serial.println("IP address: "); Serial.println(WiFi.localIP()); delay(500); +#endif // Initialize Matter EndPoint matterPref.begin("MatterPrefs", false); @@ -133,7 +142,7 @@ void setup() { Matter.begin(); // This may be a restart of a already commissioned Matter accessory if (Matter.isDeviceCommissioned()) { - Serial.println("Matter Node is commissioned and connected to Wi-Fi. Ready for use."); + Serial.println("Matter Node is commissioned and connected to the network. Ready for use."); Serial.printf( "Initial state: %s | brightness: %d | Color Temperature: %d mireds \r\n", CW_WW_Light ? "ON" : "OFF", CW_WW_Light.getBrightness(), CW_WW_Light.getColorTemperature() @@ -166,7 +175,7 @@ void loop() { ); // configure the Light based on initial on-off state and brightness CW_WW_Light.updateAccessory(); - Serial.println("Matter Node is commissioned and connected to Wi-Fi. Ready for use."); + Serial.println("Matter Node is commissioned and connected to the network. Ready for use."); } // A button is also used to control the light diff --git a/libraries/Matter/examples/MatterTemperatureSensor/MatterTemperatureSensor.ino b/libraries/Matter/examples/MatterTemperatureSensor/MatterTemperatureSensor.ino index 086155aeffe..46d6ace361f 100644 --- a/libraries/Matter/examples/MatterTemperatureSensor/MatterTemperatureSensor.ino +++ b/libraries/Matter/examples/MatterTemperatureSensor/MatterTemperatureSensor.ino @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. @@ -21,15 +21,21 @@ // Matter Manager #include +#if !CONFIG_ENABLE_CHIPOBLE +// if the device can be commissioned using BLE, WiFi is not used - save flash space #include +#endif // List of Matter Endpoints for this Node // Matter Temperature Sensor Endpoint MatterTemperatureSensor SimulatedTemperatureSensor; +// CONFIG_ENABLE_CHIPOBLE is enabled when BLE is used to commission the Matter Network +#if !CONFIG_ENABLE_CHIPOBLE // WiFi is manually set and started const char *ssid = "your-ssid"; // Change this to your WiFi SSID const char *password = "your-password"; // Change this to your WiFi password +#endif // set your board USER BUTTON pin here - decommissioning button const uint8_t buttonPin = BOOT_PIN; // Set your pin here. Using BOOT Button. @@ -60,6 +66,8 @@ void setup() { Serial.begin(115200); +// CONFIG_ENABLE_CHIPOBLE is enabled when BLE is used to commission the Matter Network +#if !CONFIG_ENABLE_CHIPOBLE // Manually connect to WiFi WiFi.begin(ssid, password); // Wait for connection @@ -68,6 +76,7 @@ void setup() { Serial.print("."); } Serial.println(); +#endif // set initial temperature sensor measurement // Simulated Sensor - it shall initially print -25C and then move to the -10C to 10C range @@ -92,7 +101,7 @@ void setup() { Serial.println("Matter Node not commissioned yet. Waiting for commissioning."); } } - Serial.println("Matter Node is commissioned and connected to Wi-Fi. Ready for use."); + Serial.println("Matter Node is commissioned and connected to the network. Ready for use."); } } diff --git a/libraries/Matter/examples/MatterThermostat/MatterThermostat.ino b/libraries/Matter/examples/MatterThermostat/MatterThermostat.ino index bf76477c846..2c446f59e34 100644 --- a/libraries/Matter/examples/MatterThermostat/MatterThermostat.ino +++ b/libraries/Matter/examples/MatterThermostat/MatterThermostat.ino @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. @@ -21,15 +21,21 @@ // Matter Manager #include +#if !CONFIG_ENABLE_CHIPOBLE +// if the device can be commissioned using BLE, WiFi is not used - save flash space #include +#endif // List of Matter Endpoints for this Node // Matter Thermostat Endpoint MatterThermostat SimulatedThermostat; +// CONFIG_ENABLE_CHIPOBLE is enabled when BLE is used to commission the Matter Network +#if !CONFIG_ENABLE_CHIPOBLE // WiFi is manually set and started const char *ssid = "your-ssid"; // Change this to your WiFi SSID const char *password = "your-password"; // Change this to your WiFi password +#endif // set your board USER BUTTON pin here - decommissioning button const uint8_t buttonPin = BOOT_PIN; // Set your pin here. Using BOOT Button. @@ -62,6 +68,8 @@ void setup() { Serial.begin(115200); +// CONFIG_ENABLE_CHIPOBLE is enabled when BLE is used to commission the Matter Network +#if !CONFIG_ENABLE_CHIPOBLE // Manually connect to WiFi WiFi.begin(ssid, password); // Wait for connection @@ -70,6 +78,7 @@ void setup() { Serial.print("."); } Serial.println(); +#endif // Simulated Thermostat in COOLING and HEATING mode with Auto Mode to keep the temperature between setpoints // Auto Mode can only be used when the control sequence of operation is Cooling & Heating @@ -94,7 +103,7 @@ void setup() { Serial.println("Matter Node not commissioned yet. Waiting for commissioning."); } } - Serial.println("Matter Node is commissioned and connected to Wi-Fi. Ready for use."); + Serial.println("Matter Node is commissioned and connected to the network. Ready for use."); // after commissioning, set initial thermostat parameters // start the thermostat in AUTO mode diff --git a/libraries/Matter/src/Matter.cpp b/libraries/Matter/src/Matter.cpp index b16edfd85c1..5ddacc1622c 100644 --- a/libraries/Matter/src/Matter.cpp +++ b/libraries/Matter/src/Matter.cpp @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. @@ -17,6 +17,10 @@ #include #include +#if CONFIG_ENABLE_MATTER_OVER_THREAD +#include "esp_openthread_types.h" +#include "platform/ESP32/OpenthreadLauncher.h" +#endif using namespace esp_matter; using namespace esp_matter::attribute; @@ -151,6 +155,18 @@ void ArduinoMatter::begin() { return; } +#if CONFIG_ENABLE_MATTER_OVER_THREAD + // Set OpenThread platform config + esp_openthread_platform_config_t config; + memset(&config, 0, sizeof(esp_openthread_platform_config_t)); + config.radio_config.radio_mode = RADIO_MODE_NATIVE; + config.host_config.host_connection_mode = HOST_CONNECTION_MODE_NONE; + config.port_config.storage_partition_name = "nvs"; + config.port_config.netif_queue_size = 10; + config.port_config.task_queue_size = 10; + set_openthread_platform_config(&config); +#endif + /* Matter start */ esp_err_t err = esp_matter::start(app_event_cb); if (err != ESP_OK) { diff --git a/libraries/Matter/src/Matter.h b/libraries/Matter/src/Matter.h index 682a0498076..09e59b4e04b 100644 --- a/libraries/Matter/src/Matter.h +++ b/libraries/Matter/src/Matter.h @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. diff --git a/libraries/Matter/src/MatterEndPoint.cpp b/libraries/Matter/src/MatterEndPoint.cpp new file mode 100644 index 00000000000..ecf1acff579 --- /dev/null +++ b/libraries/Matter/src/MatterEndPoint.cpp @@ -0,0 +1,139 @@ +// Copyright 2025 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. + +#include +#ifdef CONFIG_ESP_MATTER_ENABLE_DATA_MODEL + +#include + +uint16_t MatterEndPoint::secondary_network_endpoint_id = 0; + +// This function is called to create a secondary network interface endpoint. +// It can be used for devices that support multiple network interfaces, +// such as Ethernet, Thread and Wi-Fi. +bool MatterEndPoint::createSecondaryNetworkInterface() { + if (secondary_network_endpoint_id != 0) { + log_v("Secondary network interface endpoint already exists with ID %d", secondary_network_endpoint_id); + return false; + } + // Create a secondary network interface endpoint + endpoint::secondary_network_interface::config_t secondary_network_interface_config; + secondary_network_interface_config.network_commissioning.feature_map = chip::to_underlying( + //chip::app::Clusters::NetworkCommissioning::Feature::kWiFiNetworkInterface) | + chip::app::Clusters::NetworkCommissioning::Feature::kThreadNetworkInterface + ); + endpoint_t *endpoint = endpoint::secondary_network_interface::create(node::get(), &secondary_network_interface_config, ENDPOINT_FLAG_NONE, nullptr); + if (endpoint == nullptr) { + log_e("Failed to create secondary network interface endpoint"); + return false; + } + secondary_network_endpoint_id = endpoint::get_id(endpoint); + log_i("Secondary Network Interface created with endpoint_id %d", secondary_network_endpoint_id); + return true; +} + +uint16_t MatterEndPoint::getSecondaryNetworkEndPointId() { + return secondary_network_endpoint_id; +} + +uint16_t MatterEndPoint::getEndPointId() { + return endpoint_id; +} + +void MatterEndPoint::setEndPointId(uint16_t ep) { + if (ep == 0) { + log_e("Invalid endpoint ID"); + return; + } + log_v("Endpoint ID set to %d", ep); + + endpoint_id = ep; +} + +// helper functions for attribute manipulation +esp_matter::attribute_t *MatterEndPoint::getAttribute(uint32_t cluster_id, uint32_t attribute_id) { + if (endpoint_id == 0) { + log_e("Endpoint ID is not set"); + return nullptr; + } + endpoint_t *endpoint = endpoint::get(node::get(), endpoint_id); + if (endpoint == nullptr) { + log_e("Endpoint [%d] not found", endpoint_id); + return nullptr; + } + cluster_t *cluster = cluster::get(endpoint, cluster_id); + if (cluster == nullptr) { + log_e("Cluster [%d] not found", cluster_id); + return nullptr; + } + esp_matter::attribute_t *attribute = attribute::get(cluster, attribute_id); + if (attribute == nullptr) { + log_e("Attribute [%d] not found", attribute_id); + return nullptr; + } + return attribute; +} + +// get the value of an attribute from its cluster id and attribute it +bool MatterEndPoint::getAttributeVal(uint32_t cluster_id, uint32_t attribute_id, esp_matter_attr_val_t *attrVal) { + esp_matter::attribute_t *attribute = getAttribute(cluster_id, attribute_id); + if (attribute == nullptr) { + return false; + } + if (attribute::get_val(attribute, attrVal) == ESP_OK) { + log_v("GET_VAL Success for cluster %d, attribute %d with value %d", cluster_id, attribute_id, attrVal->val.u32); + return true; + } + log_e("GET_VAL FAILED! for cluster %d, attribute %d with value %d", cluster_id, attribute_id, attrVal->val.u32); + return false; +} + +// set the value of an attribute from its cluster id and attribute it +bool MatterEndPoint::setAttributeVal(uint32_t cluster_id, uint32_t attribute_id, esp_matter_attr_val_t *attrVal) { + esp_matter::attribute_t *attribute = getAttribute(cluster_id, attribute_id); + if (attribute == nullptr) { + return false; + } + if (attribute::set_val(attribute, attrVal) == ESP_OK) { + log_v("SET_VAL Success for cluster %d, attribute %d with value %d", cluster_id, attribute_id, attrVal->val.u32); + return true; + } + log_e("SET_VAL FAILED! for cluster %d, attribute %d with value %d", cluster_id, attribute_id, attrVal->val.u32); + return false; +} + +// update the value of an attribute from its cluster id and attribute it +bool MatterEndPoint::updateAttributeVal(uint32_t cluster_id, uint32_t attribute_id, esp_matter_attr_val_t *attrVal) { + if (attribute::update(endpoint_id, cluster_id, attribute_id, attrVal) == ESP_OK) { + log_v("Update Success for cluster %d, attribute %d with value %d", cluster_id, attribute_id, attrVal->val.u32); + return true; + } + log_e("Update FAILED! for cluster %d, attribute %d with value %d", cluster_id, attribute_id, attrVal->val.u32); + return false; +} + +// This callback is invoked when clients interact with the Identify Cluster of an specific endpoint. +bool MatterEndPoint::endpointIdentifyCB(uint16_t endpoint_id, bool identifyIsEnabled) { + if (_onEndPointIdentifyCB) { + return _onEndPointIdentifyCB(identifyIsEnabled); + } + return true; +} + +// User callback for the Identify Cluster functionality +void MatterEndPoint::onIdentify(EndPointIdentifyCB onEndPointIdentifyCB) { + _onEndPointIdentifyCB = onEndPointIdentifyCB; +} + +#endif /* CONFIG_ESP_MATTER_ENABLE_DATA_MODEL */ diff --git a/libraries/Matter/src/MatterEndPoint.h b/libraries/Matter/src/MatterEndPoint.h index 95d3d3c08df..3138014d624 100644 --- a/libraries/Matter/src/MatterEndPoint.h +++ b/libraries/Matter/src/MatterEndPoint.h @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. @@ -16,7 +16,8 @@ #include #ifdef CONFIG_ESP_MATTER_ENABLE_DATA_MODEL -#include +#include +#include #include using namespace esp_matter; @@ -29,93 +30,47 @@ class MatterEndPoint { ATTR_UPDATE = true }; - uint16_t getEndPointId() { - return endpoint_id; - } + using EndPointIdentifyCB = std::function; + + // this function is called by Matter internal event processor. It could be overwritten by the application, if necessary. + virtual bool attributeChangeCB(uint16_t endpoint_id, uint32_t cluster_id, uint32_t attribute_id, esp_matter_attr_val_t *val) = 0; + + // This function is called to create a secondary network interface endpoint. + // It can be used for devices that support multiple network interfaces, + // such as Ethernet, Thread and Wi-Fi. + bool createSecondaryNetworkInterface(); + + // This function is called to get the secondary network interface endpoint ID. + uint16_t getSecondaryNetworkEndPointId(); + + // This function is called to get the current Matter Accessory endpoint ID. + uint16_t getEndPointId(); - void setEndPointId(uint16_t ep) { - endpoint_id = ep; - } + // This function is called to set the current Matter Accessory endpoint ID. + void setEndPointId(uint16_t ep); // helper functions for attribute manipulation - esp_matter::attribute_t *getAttribute(uint32_t cluster_id, uint32_t attribute_id) { - if (endpoint_id == 0) { - log_e("Endpoint ID is not set"); - return nullptr; - } - endpoint_t *endpoint = endpoint::get(node::get(), endpoint_id); - if (endpoint == nullptr) { - log_e("Endpoint [%d] not found", endpoint_id); - return nullptr; - } - cluster_t *cluster = cluster::get(endpoint, cluster_id); - if (cluster == nullptr) { - log_e("Cluster [%d] not found", cluster_id); - return nullptr; - } - esp_matter::attribute_t *attribute = attribute::get(cluster, attribute_id); - if (attribute == nullptr) { - log_e("Attribute [%d] not found", attribute_id); - return nullptr; - } - return attribute; - } - - // get the value of an attribute from its cluster id and attribute it - bool getAttributeVal(uint32_t cluster_id, uint32_t attribute_id, esp_matter_attr_val_t *attrVal) { - esp_matter::attribute_t *attribute = getAttribute(cluster_id, attribute_id); - if (attribute == nullptr) { - return false; - } - if (attribute::get_val(attribute, attrVal) == ESP_OK) { - log_v("GET_VAL Success for cluster %d, attribute %d with value %d", cluster_id, attribute_id, attrVal->val.u32); - return true; - } - log_e("GET_VAL FAILED! for cluster %d, attribute %d with value %d", cluster_id, attribute_id, attrVal->val.u32); - return false; - } - - // set the value of an attribute from its cluster id and attribute it - bool setAttributeVal(uint32_t cluster_id, uint32_t attribute_id, esp_matter_attr_val_t *attrVal) { - esp_matter::attribute_t *attribute = getAttribute(cluster_id, attribute_id); - if (attribute == nullptr) { - return false; - } - if (attribute::set_val(attribute, attrVal) == ESP_OK) { - log_v("SET_VAL Success for cluster %d, attribute %d with value %d", cluster_id, attribute_id, attrVal->val.u32); - return true; - } - log_e("SET_VAL FAILED! for cluster %d, attribute %d with value %d", cluster_id, attribute_id, attrVal->val.u32); - return false; - } - - // update the value of an attribute from its cluster id and attribute it - bool updateAttributeVal(uint32_t cluster_id, uint32_t attribute_id, esp_matter_attr_val_t *attrVal) { - if (attribute::update(endpoint_id, cluster_id, attribute_id, attrVal) == ESP_OK) { - log_v("Update Success for cluster %d, attribute %d with value %d", cluster_id, attribute_id, attrVal->val.u32); - return true; - } - log_e("Update FAILED! for cluster %d, attribute %d with value %d", cluster_id, attribute_id, attrVal->val.u32); - return false; - } + esp_matter::attribute_t *getAttribute(uint32_t cluster_id, uint32_t attribute_id); - // this function is called by Matter internal event processor. It could be overwritten by the application, if necessary. - virtual bool attributeChangeCB(uint16_t endpoint_id, uint32_t cluster_id, uint32_t attribute_id, esp_matter_attr_val_t *val) = 0; + // get the value of an attribute from its cluster id and + bool getAttributeVal(uint32_t cluster_id, uint32_t attribute_id, esp_matter_attr_val_t *attrVal); + + // set the value of an attribute from its cluster id and + bool setAttributeVal(uint32_t cluster_id, uint32_t attribute_id, esp_matter_attr_val_t *attrVal); + + // update the value of an attribute from its cluster id + bool updateAttributeVal(uint32_t cluster_id, uint32_t attribute_id, esp_matter_attr_val_t *attrVal); // This callback is invoked when clients interact with the Identify Cluster of an specific endpoint. - bool endpointIdentifyCB(uint16_t endpoint_id, bool identifyIsEnabled) { - if (_onEndPointIdentifyCB) { - return _onEndPointIdentifyCB(identifyIsEnabled); - } - return true; - } - // User callaback for the Identify Cluster functionality - using EndPointIdentifyCB = std::function; - void onIdentify(EndPointIdentifyCB onEndPointIdentifyCB) { - _onEndPointIdentifyCB = onEndPointIdentifyCB; - } + bool endpointIdentifyCB(uint16_t endpoint_id, bool identifyIsEnabled); + + // User callback for the Identify Cluster functionality + void onIdentify(EndPointIdentifyCB onEndPointIdentifyCB); protected: + // used for secondary network interface endpoints + static uint16_t secondary_network_endpoint_id; + // main endpoint ID uint16_t endpoint_id = 0; EndPointIdentifyCB _onEndPointIdentifyCB = nullptr; }; diff --git a/libraries/Matter/src/MatterEndpoints/MatterColorLight.cpp b/libraries/Matter/src/MatterEndpoints/MatterColorLight.cpp index 39d79e86325..aff0972528b 100644 --- a/libraries/Matter/src/MatterEndpoints/MatterColorLight.cpp +++ b/libraries/Matter/src/MatterEndpoints/MatterColorLight.cpp @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. diff --git a/libraries/Matter/src/MatterEndpoints/MatterColorLight.h b/libraries/Matter/src/MatterEndpoints/MatterColorLight.h index 99449addd50..b07432a2cf5 100644 --- a/libraries/Matter/src/MatterEndpoints/MatterColorLight.h +++ b/libraries/Matter/src/MatterEndpoints/MatterColorLight.h @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. diff --git a/libraries/Matter/src/MatterEndpoints/MatterColorTemperatureLight.cpp b/libraries/Matter/src/MatterEndpoints/MatterColorTemperatureLight.cpp index 3c4fccfa046..db90edf1177 100644 --- a/libraries/Matter/src/MatterEndpoints/MatterColorTemperatureLight.cpp +++ b/libraries/Matter/src/MatterEndpoints/MatterColorTemperatureLight.cpp @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. diff --git a/libraries/Matter/src/MatterEndpoints/MatterColorTemperatureLight.h b/libraries/Matter/src/MatterEndpoints/MatterColorTemperatureLight.h index 539bc386e92..3c8b2dca882 100644 --- a/libraries/Matter/src/MatterEndpoints/MatterColorTemperatureLight.h +++ b/libraries/Matter/src/MatterEndpoints/MatterColorTemperatureLight.h @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. diff --git a/libraries/Matter/src/MatterEndpoints/MatterContactSensor.cpp b/libraries/Matter/src/MatterEndpoints/MatterContactSensor.cpp index 17b0fe7a247..52dcc8da7c0 100644 --- a/libraries/Matter/src/MatterEndpoints/MatterContactSensor.cpp +++ b/libraries/Matter/src/MatterEndpoints/MatterContactSensor.cpp @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. @@ -60,6 +60,7 @@ bool MatterContactSensor::begin(bool _contactState) { contactState = _contactState; setEndPointId(endpoint::get_id(endpoint)); log_i("Contact Sensor created with endpoint_id %d", getEndPointId()); + started = true; return true; } diff --git a/libraries/Matter/src/MatterEndpoints/MatterContactSensor.h b/libraries/Matter/src/MatterEndpoints/MatterContactSensor.h index 257da785e53..687bf0d0b4b 100644 --- a/libraries/Matter/src/MatterEndpoints/MatterContactSensor.h +++ b/libraries/Matter/src/MatterEndpoints/MatterContactSensor.h @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. diff --git a/libraries/Matter/src/MatterEndpoints/MatterDimmableLight.cpp b/libraries/Matter/src/MatterEndpoints/MatterDimmableLight.cpp index 5167cf1f21c..4659dd71675 100644 --- a/libraries/Matter/src/MatterEndpoints/MatterDimmableLight.cpp +++ b/libraries/Matter/src/MatterEndpoints/MatterDimmableLight.cpp @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. diff --git a/libraries/Matter/src/MatterEndpoints/MatterDimmableLight.h b/libraries/Matter/src/MatterEndpoints/MatterDimmableLight.h index 4497edd2fe2..45b112100cb 100644 --- a/libraries/Matter/src/MatterEndpoints/MatterDimmableLight.h +++ b/libraries/Matter/src/MatterEndpoints/MatterDimmableLight.h @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. diff --git a/libraries/Matter/src/MatterEndpoints/MatterEnhancedColorLight.cpp b/libraries/Matter/src/MatterEndpoints/MatterEnhancedColorLight.cpp index 9b245fb9408..5e9feb5aadd 100644 --- a/libraries/Matter/src/MatterEndpoints/MatterEnhancedColorLight.cpp +++ b/libraries/Matter/src/MatterEndpoints/MatterEnhancedColorLight.cpp @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. diff --git a/libraries/Matter/src/MatterEndpoints/MatterEnhancedColorLight.h b/libraries/Matter/src/MatterEndpoints/MatterEnhancedColorLight.h index a4baef968a4..97f0279b9e4 100644 --- a/libraries/Matter/src/MatterEndpoints/MatterEnhancedColorLight.h +++ b/libraries/Matter/src/MatterEndpoints/MatterEnhancedColorLight.h @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. diff --git a/libraries/Matter/src/MatterEndpoints/MatterFan.cpp b/libraries/Matter/src/MatterEndpoints/MatterFan.cpp index 1647490aa05..5745604ebde 100644 --- a/libraries/Matter/src/MatterEndpoints/MatterFan.cpp +++ b/libraries/Matter/src/MatterEndpoints/MatterFan.cpp @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. @@ -110,6 +110,7 @@ bool MatterFan::begin(uint8_t percent, FanMode_t fanMode, FanModeSequence_t fanM setEndPointId(endpoint::get_id(endpoint)); log_i("Fan created with endpoint_id %d", getEndPointId()); + started = true; return true; } diff --git a/libraries/Matter/src/MatterEndpoints/MatterFan.h b/libraries/Matter/src/MatterEndpoints/MatterFan.h index a1cd6e42423..53736ac70f2 100644 --- a/libraries/Matter/src/MatterEndpoints/MatterFan.h +++ b/libraries/Matter/src/MatterEndpoints/MatterFan.h @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. diff --git a/libraries/Matter/src/MatterEndpoints/MatterGenericSwitch.cpp b/libraries/Matter/src/MatterEndpoints/MatterGenericSwitch.cpp index e20479af088..f69d4a2ab7f 100644 --- a/libraries/Matter/src/MatterEndpoints/MatterGenericSwitch.cpp +++ b/libraries/Matter/src/MatterEndpoints/MatterGenericSwitch.cpp @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. @@ -79,6 +79,7 @@ bool MatterGenericSwitch::begin() { setEndPointId(endpoint::get_id(endpoint)); log_i("Generic Switch created with endpoint_id %d", getEndPointId()); + started = true; return true; } diff --git a/libraries/Matter/src/MatterEndpoints/MatterGenericSwitch.h b/libraries/Matter/src/MatterEndpoints/MatterGenericSwitch.h index 14118462932..ffcc6f47706 100644 --- a/libraries/Matter/src/MatterEndpoints/MatterGenericSwitch.h +++ b/libraries/Matter/src/MatterEndpoints/MatterGenericSwitch.h @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. diff --git a/libraries/Matter/src/MatterEndpoints/MatterHumiditySensor.cpp b/libraries/Matter/src/MatterEndpoints/MatterHumiditySensor.cpp index d31d0e43728..ef8276e4a04 100644 --- a/libraries/Matter/src/MatterEndpoints/MatterHumiditySensor.cpp +++ b/libraries/Matter/src/MatterEndpoints/MatterHumiditySensor.cpp @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. @@ -68,6 +68,7 @@ bool MatterHumiditySensor::begin(uint16_t _rawHumidity) { rawHumidity = _rawHumidity; setEndPointId(endpoint::get_id(endpoint)); log_i("Humidity Sensor created with endpoint_id %d", getEndPointId()); + started = true; return true; } diff --git a/libraries/Matter/src/MatterEndpoints/MatterHumiditySensor.h b/libraries/Matter/src/MatterEndpoints/MatterHumiditySensor.h index aed758b7b7a..48c3a4b9e28 100644 --- a/libraries/Matter/src/MatterEndpoints/MatterHumiditySensor.h +++ b/libraries/Matter/src/MatterEndpoints/MatterHumiditySensor.h @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. diff --git a/libraries/Matter/src/MatterEndpoints/MatterOccupancySensor.cpp b/libraries/Matter/src/MatterEndpoints/MatterOccupancySensor.cpp index 0d55c37708a..f91a02e14a9 100644 --- a/libraries/Matter/src/MatterEndpoints/MatterOccupancySensor.cpp +++ b/libraries/Matter/src/MatterEndpoints/MatterOccupancySensor.cpp @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. @@ -71,6 +71,7 @@ bool MatterOccupancySensor::begin(bool _occupancyState, OccupancySensorType_t _o occupancyState = _occupancyState; setEndPointId(endpoint::get_id(endpoint)); log_i("Occupancy Sensor created with endpoint_id %d", getEndPointId()); + started = true; return true; } diff --git a/libraries/Matter/src/MatterEndpoints/MatterOccupancySensor.h b/libraries/Matter/src/MatterEndpoints/MatterOccupancySensor.h index 30f312a9841..acfa7fec632 100644 --- a/libraries/Matter/src/MatterEndpoints/MatterOccupancySensor.h +++ b/libraries/Matter/src/MatterEndpoints/MatterOccupancySensor.h @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. diff --git a/libraries/Matter/src/MatterEndpoints/MatterOnOffLight.cpp b/libraries/Matter/src/MatterEndpoints/MatterOnOffLight.cpp index f400390f9a7..afd764088c5 100644 --- a/libraries/Matter/src/MatterEndpoints/MatterOnOffLight.cpp +++ b/libraries/Matter/src/MatterEndpoints/MatterOnOffLight.cpp @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. @@ -79,6 +79,7 @@ bool MatterOnOffLight::begin(bool initialState) { setEndPointId(endpoint::get_id(endpoint)); log_i("On-Off Light created with endpoint_id %d", getEndPointId()); + started = true; return true; } diff --git a/libraries/Matter/src/MatterEndpoints/MatterOnOffLight.h b/libraries/Matter/src/MatterEndpoints/MatterOnOffLight.h index ec524d2c300..2c52858cc9a 100644 --- a/libraries/Matter/src/MatterEndpoints/MatterOnOffLight.h +++ b/libraries/Matter/src/MatterEndpoints/MatterOnOffLight.h @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. diff --git a/libraries/Matter/src/MatterEndpoints/MatterOnOffPlugin.cpp b/libraries/Matter/src/MatterEndpoints/MatterOnOffPlugin.cpp index 9b08958684c..33cee105833 100644 --- a/libraries/Matter/src/MatterEndpoints/MatterOnOffPlugin.cpp +++ b/libraries/Matter/src/MatterEndpoints/MatterOnOffPlugin.cpp @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. @@ -78,6 +78,7 @@ bool MatterOnOffPlugin::begin(bool initialState) { onOffState = initialState; setEndPointId(endpoint::get_id(endpoint)); log_i("On-Off Plugin created with endpoint_id %d", getEndPointId()); + started = true; return true; } diff --git a/libraries/Matter/src/MatterEndpoints/MatterOnOffPlugin.h b/libraries/Matter/src/MatterEndpoints/MatterOnOffPlugin.h index 0b05c0944c4..e3489b355a8 100644 --- a/libraries/Matter/src/MatterEndpoints/MatterOnOffPlugin.h +++ b/libraries/Matter/src/MatterEndpoints/MatterOnOffPlugin.h @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. diff --git a/libraries/Matter/src/MatterEndpoints/MatterPressureSensor.cpp b/libraries/Matter/src/MatterEndpoints/MatterPressureSensor.cpp index 86d245d4041..8674ad5936b 100644 --- a/libraries/Matter/src/MatterEndpoints/MatterPressureSensor.cpp +++ b/libraries/Matter/src/MatterEndpoints/MatterPressureSensor.cpp @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. @@ -61,6 +61,7 @@ bool MatterPressureSensor::begin(int16_t _rawPressure) { rawPressure = _rawPressure; setEndPointId(endpoint::get_id(endpoint)); log_i("Pressure Sensor created with endpoint_id %d", getEndPointId()); + started = true; return true; } diff --git a/libraries/Matter/src/MatterEndpoints/MatterPressureSensor.h b/libraries/Matter/src/MatterEndpoints/MatterPressureSensor.h index 0715c05609d..6b1973049ca 100644 --- a/libraries/Matter/src/MatterEndpoints/MatterPressureSensor.h +++ b/libraries/Matter/src/MatterEndpoints/MatterPressureSensor.h @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. diff --git a/libraries/Matter/src/MatterEndpoints/MatterTemperatureSensor.cpp b/libraries/Matter/src/MatterEndpoints/MatterTemperatureSensor.cpp index 863f86386c1..248bd29f05f 100644 --- a/libraries/Matter/src/MatterEndpoints/MatterTemperatureSensor.cpp +++ b/libraries/Matter/src/MatterEndpoints/MatterTemperatureSensor.cpp @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. @@ -61,6 +61,7 @@ bool MatterTemperatureSensor::begin(int16_t _rawTemperature) { rawTemperature = _rawTemperature; setEndPointId(endpoint::get_id(endpoint)); log_i("Temperature Sensor created with endpoint_id %d", getEndPointId()); + started = true; return true; } diff --git a/libraries/Matter/src/MatterEndpoints/MatterTemperatureSensor.h b/libraries/Matter/src/MatterEndpoints/MatterTemperatureSensor.h index 3be6101166c..51a1eeb3b0a 100644 --- a/libraries/Matter/src/MatterEndpoints/MatterTemperatureSensor.h +++ b/libraries/Matter/src/MatterEndpoints/MatterTemperatureSensor.h @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. diff --git a/libraries/Matter/src/MatterEndpoints/MatterThermostat.cpp b/libraries/Matter/src/MatterEndpoints/MatterThermostat.cpp index 5a68421bd8a..6ee18ef0cc9 100644 --- a/libraries/Matter/src/MatterEndpoints/MatterThermostat.cpp +++ b/libraries/Matter/src/MatterEndpoints/MatterThermostat.cpp @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. @@ -202,6 +202,7 @@ bool MatterThermostat::begin(ControlSequenceOfOperation_t _controlSequence, Ther setEndPointId(endpoint::get_id(endpoint)); log_i("Thermostat created with endpoint_id %d", getEndPointId()); + started = true; return true; } diff --git a/libraries/Matter/src/MatterEndpoints/MatterThermostat.h b/libraries/Matter/src/MatterEndpoints/MatterThermostat.h index 2d64bdf3b01..53df61d191e 100644 --- a/libraries/Matter/src/MatterEndpoints/MatterThermostat.h +++ b/libraries/Matter/src/MatterEndpoints/MatterThermostat.h @@ -1,4 +1,4 @@ -// Copyright 2024 Espressif Systems (Shanghai) PTE LTD +// Copyright 2025 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. From 18f647611bda1a5e0f7fc5bb95a21c40f18e0f74 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jan=20Proch=C3=A1zka?= <90197375+P-R-O-C-H-Y@users.noreply.github.com> Date: Thu, 3 Jul 2025 15:39:46 +0200 Subject: [PATCH 091/173] fix(spi): Fix bus clock for ESP32-P4 + remove S2 leftover (#11547) * fix(spi): Fix bus clock for ESP32-P4 * fix(ci): Ignore unused-variable warning * ci(pre-commit): Apply automatic fixes --------- Co-authored-by: pre-commit-ci-lite[bot] <117423508+pre-commit-ci-lite[bot]@users.noreply.github.com> --- cores/esp32/esp32-hal-spi.c | 29 ++++++++++++++++++++++++++--- 1 file changed, 26 insertions(+), 3 deletions(-) diff --git a/cores/esp32/esp32-hal-spi.c b/cores/esp32/esp32-hal-spi.c index d39aceb5f8d..107b94da0d6 100644 --- a/cores/esp32/esp32-hal-spi.c +++ b/cores/esp32/esp32-hal-spi.c @@ -60,6 +60,7 @@ #elif CONFIG_IDF_TARGET_ESP32P4 #include "esp32p4/rom/ets_sys.h" #include "esp32p4/rom/gpio.h" +#include "hal/spi_ll.h" #else #error Target CONFIG_IDF_TARGET is not supported #endif @@ -639,9 +640,6 @@ spi_t *spiStartBus(uint8_t spi_num, uint32_t clockDiv, uint8_t dataMode, uint8_t } else if (spi_num == HSPI) { DPORT_SET_PERI_REG_MASK(DPORT_PERIP_CLK_EN_REG, DPORT_SPI3_CLK_EN); DPORT_CLEAR_PERI_REG_MASK(DPORT_PERIP_RST_EN_REG, DPORT_SPI3_RST); - } else { - DPORT_SET_PERI_REG_MASK(DPORT_PERIP_CLK_EN_REG, DPORT_SPI01_CLK_EN); - DPORT_CLEAR_PERI_REG_MASK(DPORT_PERIP_RST_EN_REG, DPORT_SPI01_RST); } #elif CONFIG_IDF_TARGET_ESP32S3 if (spi_num == FSPI) { @@ -662,6 +660,31 @@ spi_t *spiStartBus(uint8_t spi_num, uint32_t clockDiv, uint8_t dataMode, uint8_t DPORT_SET_PERI_REG_MASK(DPORT_PERIP_CLK_EN_REG, DPORT_SPI01_CLK_EN); DPORT_CLEAR_PERI_REG_MASK(DPORT_PERIP_RST_EN_REG, DPORT_SPI01_RST); } +#elif CONFIG_IDF_TARGET_ESP32P4 +#pragma GCC diagnostic push +#pragma GCC diagnostic ignored "-Wunused-variable" + if (spi_num == FSPI) { + PERIPH_RCC_ACQUIRE_ATOMIC(PERIPH_GPSPI2_MODULE, ref_count) { + if (ref_count == 0) { + PERIPH_RCC_ATOMIC() { + spi_ll_enable_bus_clock(SPI2_HOST, true); + spi_ll_reset_register(SPI2_HOST); + spi_ll_enable_clock(SPI2_HOST, true); + } + } + } + } else if (spi_num == HSPI) { + PERIPH_RCC_ACQUIRE_ATOMIC(PERIPH_GPSPI3_MODULE, ref_count) { + if (ref_count == 0) { + PERIPH_RCC_ATOMIC() { + spi_ll_enable_bus_clock(SPI3_HOST, true); + spi_ll_reset_register(SPI3_HOST); + spi_ll_enable_clock(SPI3_HOST, true); + } + } + } + } +#pragma GCC diagnostic pop #elif defined(__PERIPH_CTRL_ALLOW_LEGACY_API) periph_ll_reset(PERIPH_SPI2_MODULE); periph_ll_enable_clk_clear_rst(PERIPH_SPI2_MODULE); From ac961f671abd5ae1da0a15fd4bee71ed807c2cf3 Mon Sep 17 00:00:00 2001 From: me-no-dev Date: Thu, 3 Jul 2025 16:42:28 +0300 Subject: [PATCH 092/173] Update core version to 3.2.1 --- cores/esp32/esp_arduino_version.h | 2 +- libraries/ArduinoOTA/library.properties | 2 +- libraries/AsyncUDP/library.properties | 2 +- libraries/BLE/library.properties | 2 +- libraries/BluetoothSerial/library.properties | 2 +- libraries/DNSServer/library.properties | 2 +- libraries/EEPROM/library.properties | 2 +- libraries/ESP32/library.properties | 2 +- libraries/ESP_I2S/library.properties | 2 +- libraries/ESP_NOW/library.properties | 2 +- libraries/ESP_SR/library.properties | 2 +- libraries/ESPmDNS/library.properties | 2 +- libraries/Ethernet/library.properties | 2 +- libraries/FFat/library.properties | 2 +- libraries/FS/library.properties | 2 +- libraries/HTTPClient/library.properties | 2 +- libraries/HTTPUpdate/library.properties | 2 +- libraries/HTTPUpdateServer/library.properties | 2 +- libraries/Insights/library.properties | 2 +- libraries/LittleFS/library.properties | 2 +- libraries/Matter/library.properties | 2 +- libraries/NetBIOS/library.properties | 2 +- libraries/Network/library.properties | 2 +- libraries/NetworkClientSecure/library.properties | 2 +- libraries/OpenThread/library.properties | 2 +- libraries/PPP/library.properties | 2 +- libraries/Preferences/library.properties | 2 +- libraries/RainMaker/library.properties | 2 +- libraries/SD/library.properties | 2 +- libraries/SD_MMC/library.properties | 2 +- libraries/SPI/library.properties | 2 +- libraries/SPIFFS/library.properties | 2 +- libraries/SimpleBLE/library.properties | 2 +- libraries/TFLiteMicro/library.properties | 2 +- libraries/Ticker/library.properties | 2 +- libraries/USB/library.properties | 2 +- libraries/Update/library.properties | 2 +- libraries/WebServer/library.properties | 2 +- libraries/WiFi/library.properties | 2 +- libraries/WiFiProv/library.properties | 2 +- libraries/Wire/library.properties | 2 +- libraries/Zigbee/library.properties | 2 +- package.json | 2 +- platform.txt | 2 +- 44 files changed, 44 insertions(+), 44 deletions(-) diff --git a/cores/esp32/esp_arduino_version.h b/cores/esp32/esp_arduino_version.h index b1355e908ae..97bb3ac794b 100644 --- a/cores/esp32/esp_arduino_version.h +++ b/cores/esp32/esp_arduino_version.h @@ -23,7 +23,7 @@ extern "C" { /** Minor version number (x.X.x) */ #define ESP_ARDUINO_VERSION_MINOR 2 /** Patch version number (x.x.X) */ -#define ESP_ARDUINO_VERSION_PATCH 0 +#define ESP_ARDUINO_VERSION_PATCH 1 /** * Macro to convert ARDUINO version number into an integer diff --git a/libraries/ArduinoOTA/library.properties b/libraries/ArduinoOTA/library.properties index 0796eddf318..18de5aa2180 100644 --- a/libraries/ArduinoOTA/library.properties +++ b/libraries/ArduinoOTA/library.properties @@ -1,5 +1,5 @@ name=ArduinoOTA -version=3.2.0 +version=3.2.1 author=Ivan Grokhotkov and Hristo Gochkov maintainer=Hristo Gochkov sentence=Enables Over The Air upgrades, via wifi and espota.py UDP request/TCP download. diff --git a/libraries/AsyncUDP/library.properties b/libraries/AsyncUDP/library.properties index 116dcbacaa8..c64c60d0421 100644 --- a/libraries/AsyncUDP/library.properties +++ b/libraries/AsyncUDP/library.properties @@ -1,5 +1,5 @@ name=ESP32 Async UDP -version=3.2.0 +version=3.2.1 author=Me-No-Dev maintainer=Me-No-Dev sentence=Async UDP Library for ESP32 diff --git a/libraries/BLE/library.properties b/libraries/BLE/library.properties index 7ef636223ec..b009b14e194 100644 --- a/libraries/BLE/library.properties +++ b/libraries/BLE/library.properties @@ -1,5 +1,5 @@ name=BLE -version=3.2.0 +version=3.2.1 author=Neil Kolban maintainer=Dariusz Krempa sentence=BLE functions for ESP32 diff --git a/libraries/BluetoothSerial/library.properties b/libraries/BluetoothSerial/library.properties index 0a382410bba..4bc1427e3f8 100644 --- a/libraries/BluetoothSerial/library.properties +++ b/libraries/BluetoothSerial/library.properties @@ -1,5 +1,5 @@ name=BluetoothSerial -version=3.2.0 +version=3.2.1 author=Evandro Copercini maintainer=Evandro Copercini sentence=Simple UART to Classical Bluetooth bridge for ESP32 diff --git a/libraries/DNSServer/library.properties b/libraries/DNSServer/library.properties index 5e70a6ec03a..42e4c38dc9d 100644 --- a/libraries/DNSServer/library.properties +++ b/libraries/DNSServer/library.properties @@ -1,5 +1,5 @@ name=DNSServer -version=3.2.0 +version=3.2.1 author=Kristijan Novoselić maintainer=Kristijan Novoselić, sentence=A simple DNS server for ESP32. diff --git a/libraries/EEPROM/library.properties b/libraries/EEPROM/library.properties index c7e48501c04..ee1caae0792 100644 --- a/libraries/EEPROM/library.properties +++ b/libraries/EEPROM/library.properties @@ -1,5 +1,5 @@ name=EEPROM -version=3.2.0 +version=3.2.1 author=Ivan Grokhotkov maintainer=Paolo Becchi sentence=Enables reading and writing data a sequential, addressable FLASH storage diff --git a/libraries/ESP32/library.properties b/libraries/ESP32/library.properties index 7ebc69be71f..0815a69ce6b 100644 --- a/libraries/ESP32/library.properties +++ b/libraries/ESP32/library.properties @@ -1,5 +1,5 @@ name=ESP32 -version=3.2.0 +version=3.2.1 author=Hristo Gochkov, Ivan Grokhtkov maintainer=Hristo Gochkov sentence=ESP32 sketches examples diff --git a/libraries/ESP_I2S/library.properties b/libraries/ESP_I2S/library.properties index 263e9823275..ff1ad5d59da 100644 --- a/libraries/ESP_I2S/library.properties +++ b/libraries/ESP_I2S/library.properties @@ -1,5 +1,5 @@ name=ESP_I2S -version=3.2.0 +version=3.2.1 author=me-no-dev maintainer=me-no-dev sentence=Library for ESP I2S communication diff --git a/libraries/ESP_NOW/library.properties b/libraries/ESP_NOW/library.properties index f3e5c109a9b..426a9464ace 100644 --- a/libraries/ESP_NOW/library.properties +++ b/libraries/ESP_NOW/library.properties @@ -1,5 +1,5 @@ name=ESP_NOW -version=3.2.0 +version=3.2.1 author=me-no-dev maintainer=P-R-O-C-H-Y sentence=Library for ESP_NOW diff --git a/libraries/ESP_SR/library.properties b/libraries/ESP_SR/library.properties index 295761bd9fb..3b9777bca8f 100644 --- a/libraries/ESP_SR/library.properties +++ b/libraries/ESP_SR/library.properties @@ -1,5 +1,5 @@ name=ESP_SR -version=3.2.0 +version=3.2.1 author=me-no-dev maintainer=me-no-dev sentence=Library for ESP Sound Recognition diff --git a/libraries/ESPmDNS/library.properties b/libraries/ESPmDNS/library.properties index 6d36d61b783..b99a5f58e5c 100644 --- a/libraries/ESPmDNS/library.properties +++ b/libraries/ESPmDNS/library.properties @@ -1,5 +1,5 @@ name=ESPmDNS -version=3.2.0 +version=3.2.1 author=Hristo Gochkov, Ivan Grokhtkov maintainer=Hristo Gochkov sentence=ESP32 mDNS Library diff --git a/libraries/Ethernet/library.properties b/libraries/Ethernet/library.properties index d34ae036417..cd2d8ead018 100644 --- a/libraries/Ethernet/library.properties +++ b/libraries/Ethernet/library.properties @@ -1,5 +1,5 @@ name=Ethernet -version=3.2.0 +version=3.2.1 author=Hristo Gochkov maintainer=Hristo Gochkov sentence=Enables network connection (local and Internet) using the ESP32 Ethernet. diff --git a/libraries/FFat/library.properties b/libraries/FFat/library.properties index 35940fd5472..898982536b0 100644 --- a/libraries/FFat/library.properties +++ b/libraries/FFat/library.properties @@ -1,5 +1,5 @@ name=FFat -version=3.2.0 +version=3.2.1 author=Hristo Gochkov, Ivan Grokhtkov, Larry Bernstone maintainer=Hristo Gochkov sentence=ESP32 FAT on Flash File System diff --git a/libraries/FS/library.properties b/libraries/FS/library.properties index 07bd296bb83..fe86d613bdf 100644 --- a/libraries/FS/library.properties +++ b/libraries/FS/library.properties @@ -1,5 +1,5 @@ name=FS -version=3.2.0 +version=3.2.1 author=Hristo Gochkov, Ivan Grokhtkov maintainer=Hristo Gochkov sentence=ESP32 File System diff --git a/libraries/HTTPClient/library.properties b/libraries/HTTPClient/library.properties index f2dafc36d1b..76da4c857e1 100644 --- a/libraries/HTTPClient/library.properties +++ b/libraries/HTTPClient/library.properties @@ -1,5 +1,5 @@ name=HTTPClient -version=3.2.0 +version=3.2.1 author=Markus Sattler maintainer=Markus Sattler sentence=HTTP Client for ESP32 diff --git a/libraries/HTTPUpdate/library.properties b/libraries/HTTPUpdate/library.properties index 419f3b97b3f..dfe0474c3f3 100644 --- a/libraries/HTTPUpdate/library.properties +++ b/libraries/HTTPUpdate/library.properties @@ -1,5 +1,5 @@ name=HTTPUpdate -version=3.2.0 +version=3.2.1 author=Markus Sattler maintainer=Markus Sattler sentence=Http Update for ESP32 diff --git a/libraries/HTTPUpdateServer/library.properties b/libraries/HTTPUpdateServer/library.properties index 9c793a26ac8..90aa966d27f 100644 --- a/libraries/HTTPUpdateServer/library.properties +++ b/libraries/HTTPUpdateServer/library.properties @@ -1,5 +1,5 @@ name=HTTPUpdateServer -version=3.2.0 +version=3.2.1 author=Hristo Kapanakov maintainer= sentence=Simple HTTP Update server based on the WebServer diff --git a/libraries/Insights/library.properties b/libraries/Insights/library.properties index fefe5aab177..c1ad9c72ce1 100644 --- a/libraries/Insights/library.properties +++ b/libraries/Insights/library.properties @@ -1,5 +1,5 @@ name=ESP Insights -version=3.2.0 +version=3.2.1 author=Sanket Wadekar maintainer=Sanket Wadekar sentence=ESP Insights diff --git a/libraries/LittleFS/library.properties b/libraries/LittleFS/library.properties index a9dae69b7f8..e00ed49c312 100644 --- a/libraries/LittleFS/library.properties +++ b/libraries/LittleFS/library.properties @@ -1,5 +1,5 @@ name=LittleFS -version=3.2.0 +version=3.2.1 author= maintainer= sentence=LittleFS for esp32 diff --git a/libraries/Matter/library.properties b/libraries/Matter/library.properties index ac9e0964ab5..b601fce0ff5 100644 --- a/libraries/Matter/library.properties +++ b/libraries/Matter/library.properties @@ -1,5 +1,5 @@ name=Matter -version=3.2.0 +version=3.2.1 author=Rodrigo Garcia | GitHub @SuGlider maintainer=Rodrigo Garcia sentence=Library for supporting Matter environment on ESP32. diff --git a/libraries/NetBIOS/library.properties b/libraries/NetBIOS/library.properties index 5f134bfdc55..a4bbf93c0ed 100644 --- a/libraries/NetBIOS/library.properties +++ b/libraries/NetBIOS/library.properties @@ -1,5 +1,5 @@ name=NetBIOS -version=3.2.0 +version=3.2.1 author=Pablo@xpablo.cz maintainer=Hristo Gochkov sentence=Enables NBNS (NetBIOS) name resolution. diff --git a/libraries/Network/library.properties b/libraries/Network/library.properties index 0b821e08d77..f2284981704 100644 --- a/libraries/Network/library.properties +++ b/libraries/Network/library.properties @@ -1,5 +1,5 @@ name=Networking -version=3.2.0 +version=3.2.1 author=Hristo Gochkov maintainer=Hristo Gochkov sentence=General network management library. diff --git a/libraries/NetworkClientSecure/library.properties b/libraries/NetworkClientSecure/library.properties index 455dea6a2bf..31af7a1bc8c 100644 --- a/libraries/NetworkClientSecure/library.properties +++ b/libraries/NetworkClientSecure/library.properties @@ -1,5 +1,5 @@ name=NetworkClientSecure -version=3.2.0 +version=3.2.1 author=Evandro Luis Copercini maintainer=Github Community sentence=Enables secure network connection (local and Internet) using the ESP32 built-in WiFi. diff --git a/libraries/OpenThread/library.properties b/libraries/OpenThread/library.properties index 0e547d188aa..2687b1dcd1c 100644 --- a/libraries/OpenThread/library.properties +++ b/libraries/OpenThread/library.properties @@ -1,5 +1,5 @@ name=OpenThread -version=3.2.0 +version=3.2.1 author=Rodrigo Garcia | GitHub @SuGlider maintainer=Rodrigo Garcia sentence=Library for OpenThread Network on ESP32. diff --git a/libraries/PPP/library.properties b/libraries/PPP/library.properties index 7158a027b0a..c39647cbe56 100644 --- a/libraries/PPP/library.properties +++ b/libraries/PPP/library.properties @@ -1,5 +1,5 @@ name=PPP -version=3.2.0 +version=3.2.1 author=Hristo Gochkov maintainer=Hristo Gochkov sentence=Enables network connection using GSM Modem. diff --git a/libraries/Preferences/library.properties b/libraries/Preferences/library.properties index eb0158e4932..6e0d38348c0 100644 --- a/libraries/Preferences/library.properties +++ b/libraries/Preferences/library.properties @@ -1,5 +1,5 @@ name=Preferences -version=3.2.0 +version=3.2.1 author=Hristo Gochkov maintainer=Hristo Gochkov sentence=Provides friendly access to ESP32's Non-Volatile Storage diff --git a/libraries/RainMaker/library.properties b/libraries/RainMaker/library.properties index 95ce14d6708..71b4082a0a7 100644 --- a/libraries/RainMaker/library.properties +++ b/libraries/RainMaker/library.properties @@ -1,5 +1,5 @@ name=ESP RainMaker -version=3.2.0 +version=3.2.1 author=Sweety Mhaiske maintainer=Hristo Gochkov sentence=ESP RainMaker Support diff --git a/libraries/SD/library.properties b/libraries/SD/library.properties index 66c4f5cfafd..cc51196ed54 100644 --- a/libraries/SD/library.properties +++ b/libraries/SD/library.properties @@ -1,5 +1,5 @@ name=SD -version=3.2.0 +version=3.2.1 author=Arduino, SparkFun maintainer=Arduino sentence=Enables reading and writing on SD cards. For all Arduino boards. diff --git a/libraries/SD_MMC/library.properties b/libraries/SD_MMC/library.properties index 855390e5057..590eb8ebc52 100644 --- a/libraries/SD_MMC/library.properties +++ b/libraries/SD_MMC/library.properties @@ -1,5 +1,5 @@ name=SD_MMC -version=3.2.0 +version=3.2.1 author=Hristo Gochkov, Ivan Grokhtkov maintainer=Hristo Gochkov sentence=ESP32 SDMMC File System diff --git a/libraries/SPI/library.properties b/libraries/SPI/library.properties index 64db93aceeb..724137030d1 100644 --- a/libraries/SPI/library.properties +++ b/libraries/SPI/library.properties @@ -1,5 +1,5 @@ name=SPI -version=3.2.0 +version=3.2.1 author=Hristo Gochkov maintainer=Hristo Gochkov sentence=Enables the communication with devices that use the Serial Peripheral Interface (SPI) Bus. For all Arduino boards, BUT Arduino DUE. diff --git a/libraries/SPIFFS/library.properties b/libraries/SPIFFS/library.properties index 78f77fe9794..fc4601e512c 100644 --- a/libraries/SPIFFS/library.properties +++ b/libraries/SPIFFS/library.properties @@ -1,5 +1,5 @@ name=SPIFFS -version=3.2.0 +version=3.2.1 author=Hristo Gochkov, Ivan Grokhtkov maintainer=Hristo Gochkov sentence=ESP32 SPIFFS File System diff --git a/libraries/SimpleBLE/library.properties b/libraries/SimpleBLE/library.properties index ad5e10d3acb..768449ee1c4 100644 --- a/libraries/SimpleBLE/library.properties +++ b/libraries/SimpleBLE/library.properties @@ -1,5 +1,5 @@ name=SimpleBLE -version=3.2.0 +version=3.2.1 author=Hristo Gochkov maintainer=Hristo Gochkov sentence=Provides really simple BLE advertizer with just on and off diff --git a/libraries/TFLiteMicro/library.properties b/libraries/TFLiteMicro/library.properties index 1e8db045610..6ad0c32c7d5 100644 --- a/libraries/TFLiteMicro/library.properties +++ b/libraries/TFLiteMicro/library.properties @@ -1,5 +1,5 @@ name=TFLite Micro -version=3.2.0 +version=3.2.1 author=Sanket Wadekar maintainer=Sanket Wadekar sentence=TensorFlow Lite for Microcontrollers diff --git a/libraries/Ticker/library.properties b/libraries/Ticker/library.properties index 975db96d1ad..8a2af554906 100644 --- a/libraries/Ticker/library.properties +++ b/libraries/Ticker/library.properties @@ -1,5 +1,5 @@ name=Ticker -version=3.2.0 +version=3.2.1 author=Bert Melis maintainer=Hristo Gochkov sentence=Allows to call functions with a given interval. diff --git a/libraries/USB/library.properties b/libraries/USB/library.properties index 9d47dfc6719..677d736a635 100644 --- a/libraries/USB/library.properties +++ b/libraries/USB/library.properties @@ -1,5 +1,5 @@ name=USB -version=3.2.0 +version=3.2.1 author=Hristo Gochkov maintainer=Hristo Gochkov sentence=ESP32S2 USB Library diff --git a/libraries/Update/library.properties b/libraries/Update/library.properties index c3ee8f7e506..4c756397aba 100644 --- a/libraries/Update/library.properties +++ b/libraries/Update/library.properties @@ -1,5 +1,5 @@ name=Update -version=3.2.0 +version=3.2.1 author=Hristo Gochkov maintainer=Hristo Gochkov sentence=ESP32 Sketch Update Library diff --git a/libraries/WebServer/library.properties b/libraries/WebServer/library.properties index 2a9ff530d57..bf6c26c65e7 100644 --- a/libraries/WebServer/library.properties +++ b/libraries/WebServer/library.properties @@ -1,5 +1,5 @@ name=WebServer -version=3.2.0 +version=3.2.1 author=Ivan Grokhotkov maintainer=Ivan Grokhtkov sentence=Simple web server library diff --git a/libraries/WiFi/library.properties b/libraries/WiFi/library.properties index 03112c2fcc6..a282570ff8a 100644 --- a/libraries/WiFi/library.properties +++ b/libraries/WiFi/library.properties @@ -1,5 +1,5 @@ name=WiFi -version=3.2.0 +version=3.2.1 author=Hristo Gochkov maintainer=Hristo Gochkov sentence=Enables network connection (local and Internet) using the ESP32 built-in WiFi. diff --git a/libraries/WiFiProv/library.properties b/libraries/WiFiProv/library.properties index 13a63c50bb1..1cc8e4b0f91 100644 --- a/libraries/WiFiProv/library.properties +++ b/libraries/WiFiProv/library.properties @@ -1,5 +1,5 @@ name=WiFiProv -version=3.2.0 +version=3.2.1 author=Switi Mhaiske maintainer=Hristo Gochkov sentence=Enables provisioning. diff --git a/libraries/Wire/library.properties b/libraries/Wire/library.properties index 655f4bd3194..22cc7f26d86 100644 --- a/libraries/Wire/library.properties +++ b/libraries/Wire/library.properties @@ -1,5 +1,5 @@ name=Wire -version=3.2.0 +version=3.2.1 author=Hristo Gochkov maintainer=Hristo Gochkov sentence=Allows the communication between devices or sensors connected via Two Wire Interface Bus. For esp8266 boards. diff --git a/libraries/Zigbee/library.properties b/libraries/Zigbee/library.properties index 9a558d70216..d9587f49fd5 100644 --- a/libraries/Zigbee/library.properties +++ b/libraries/Zigbee/library.properties @@ -1,5 +1,5 @@ name=Zigbee -version=3.2.0 +version=3.2.1 author=P-R-O-C-H-Y maintainer=Jan Procházka sentence=Enables zigbee connection with the ESP32 diff --git a/package.json b/package.json index 9c918733209..85a15ab3615 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "framework-arduinoespressif32", - "version": "3.2.0", + "version": "3.2.1", "description": "Arduino Wiring-based Framework for the Espressif ESP32, ESP32-P4, ESP32-S and ESP32-C series of SoCs", "keywords": [ "framework", diff --git a/platform.txt b/platform.txt index 7bc89426323..62cb829dcf0 100644 --- a/platform.txt +++ b/platform.txt @@ -1,5 +1,5 @@ name=ESP32 Arduino -version=3.2.0 +version=3.2.1 tools.esp32-arduino-libs.path={runtime.platform.path}/tools/esp32-arduino-libs tools.esp32-arduino-libs.path.windows={runtime.platform.path}\tools\esp32-arduino-libs From 0b9c9362deb4d24953a4a5b1196915e551e7e860 Mon Sep 17 00:00:00 2001 From: Dogus Cendek Date: Thu, 3 Jul 2025 23:13:42 +0300 Subject: [PATCH 093/173] Add Deneyap Kart v2 (#11545) * Updated Pins of Devkits Deleted soc_caps.h library and related commands at Deneyap Kart 1A v2, Deneyap Kart 1A, Deneyap Mini and Deneyap Mini v2. Added TX1 and RX1 pins and updated LED pin definition at all Devkits. Added BOOT (BT) pins at Deneyap Kart, Deneyap Kart 1A, Deneyap Mini and Deneyap Kart G. Changed D0 and D1 pin numbers at Deneyap Kart G. Changed D12, D13, D14, D15, PWM0 and PWM1 pin numbers at Deneyap Kart 1A v2. Added A8, T0, T1, T2, T3, T4, T5, T6, T7, T8, D16, D17, D18, D19, PWM2, PWM3, PWM4 and BAT pin numbers at Deneyap Kart 1A v2. Changed A2, A3, A4 (T0) and A5 (T1) pin numbers at Deneyap Kart and Deneyap Kart 1A. Renamed DA2 (DAC2) pin as DA0 (DAC0) and changed DAC1 and DAC2 pin numbers at Deneyap Mini and Deneyap Mini v2. * Updated board.txt of all Devkits Updated board.txt of all Devkits * Remove Repeating Pin Definition Remove Repeating Pin Definition * Fix Pin Definition Remove repeating pin definitions of SPI, I2C and DAC. Update RGB LED definition for using digitalWrite() command with RGB LED. * Remove Repeating Pin Definitions Remove repeating pin definitions of LEDB, SPI, I2C and DAC. * Update RGB LED definition Update RGB LED definition for using digitalWrite() command with RGB LED. * Fix broken links for external library test Fix broken links for external library test * Update UploadMode Config of Deneyap Kart 1A v2 Update UploadMode Config of Deneyap Kart 1A v2 * Add Deneyap Kart v2 Add pin definitions and configs of Deneyap Kart v2. * Update UploadMode config Hardware CDC is default now. * Fixed typo fault Fixed typo fault * Fixed build.board parameter Fixed build.board parameter * Removed unsupported Flash sizes and RAM type Removed unsupported Flash sizes and RAM type from menu. * Remove unsupported partition options Remove unsupported partition options * Fixed Annotations and Space * Update pins_arduino.h --- boards.txt | 223 ++++++++++++++++++++++++++ variants/deneyapkartv2/pins_arduino.h | 123 ++++++++++++++ 2 files changed, 346 insertions(+) create mode 100644 variants/deneyapkartv2/pins_arduino.h diff --git a/boards.txt b/boards.txt index 4198f3856b0..e47995985ed 100644 --- a/boards.txt +++ b/boards.txt @@ -33287,6 +33287,229 @@ deneyapkart.menu.EraseFlash.all.upload.erase_cmd=-e ############################################################## +deneyapkartv2.name=Deneyap Kart v2 + +deneyapkartv2.vid.0=0x303a +deneyapkartv2.pid.0=0x82EB + +deneyapkartv2.bootloader.tool=esptool_py +deneyapkartv2.bootloader.tool.default=esptool_py + +deneyapkartv2.upload.tool=esptool_py +deneyapkartv2.upload.tool.default=esptool_py +deneyapkartv2.upload.tool.network=esp_ota + +deneyapkartv2.upload.maximum_size=1310720 +deneyapkartv2.upload.maximum_data_size=327680 +deneyapkartv2.upload.flags= +deneyapkartv2.upload.extra_flags= +deneyapkartv2.upload.use_1200bps_touch=false +deneyapkartv2.upload.wait_for_upload_port=false + +deneyapkartv2.serial.disableDTR=false +deneyapkartv2.serial.disableRTS=false + +deneyapkartv2.build.tarch=xtensa +deneyapkartv2.build.bootloader_addr=0x0 +deneyapkartv2.build.target=esp32s3 +deneyapkartv2.build.mcu=esp32s3 +deneyapkartv2.build.core=esp32 +deneyapkartv2.build.variant=deneyapkartv2 +deneyapkartv2.build.board=DYDKV2 + +deneyapkartv2.build.usb_mode=1 +deneyapkartv2.build.cdc_on_boot=1 +deneyapkartv2.build.msc_on_boot=0 +deneyapkartv2.build.dfu_on_boot=0 +deneyapkartv2.build.f_cpu=240000000L +deneyapkartv2.build.flash_size=4MB +deneyapkartv2.build.flash_freq=80m +deneyapkartv2.build.flash_mode=dio +deneyapkartv2.build.boot=qio +deneyapkartv2.build.boot_freq=80m +deneyapkartv2.build.partitions=default +deneyapkartv2.build.defines=-DBOARD_HAS_PSRAM +deneyapkartv2.build.loop_core= +deneyapkartv2.build.event_core= +deneyapkartv2.build.psram_type=opi +deneyapkartv2.build.memory_type={build.boot}_{build.psram_type} + +## IDE 2.0 Seems to not update the value +deneyapkartv2.menu.JTAGAdapter.default=Disabled +deneyapkartv2.menu.JTAGAdapter.default.build.copy_jtag_files=0 +deneyapkartv2.menu.JTAGAdapter.builtin=Integrated USB JTAG +deneyapkartv2.menu.JTAGAdapter.builtin.build.openocdscript=esp32s3-builtin.cfg +deneyapkartv2.menu.JTAGAdapter.builtin.build.copy_jtag_files=1 +deneyapkartv2.menu.JTAGAdapter.external=FTDI Adapter +deneyapkartv2.menu.JTAGAdapter.external.build.openocdscript=esp32s3-ftdi.cfg +deneyapkartv2.menu.JTAGAdapter.external.build.copy_jtag_files=1 +deneyapkartv2.menu.JTAGAdapter.bridge=ESP USB Bridge +deneyapkartv2.menu.JTAGAdapter.bridge.build.openocdscript=esp32s3-bridge.cfg +deneyapkartv2.menu.JTAGAdapter.bridge.build.copy_jtag_files=1 + +deneyapkartv2.menu.PSRAM.opi=OPI PSRAM +deneyapkartv2.menu.PSRAM.opi.build.defines=-DBOARD_HAS_PSRAM +deneyapkartv2.menu.PSRAM.opi.build.psram_type=opi +deneyapkartv2.menu.PSRAM.disabled=Disabled +deneyapkartv2.menu.PSRAM.disabled.build.defines= +deneyapkartv2.menu.PSRAM.disabled.build.psram_type=qspi + +deneyapkartv2.menu.FlashMode.qio=QIO 80MHz +deneyapkartv2.menu.FlashMode.qio.build.flash_mode=dio +deneyapkartv2.menu.FlashMode.qio.build.boot=qio +deneyapkartv2.menu.FlashMode.qio.build.boot_freq=80m +deneyapkartv2.menu.FlashMode.qio.build.flash_freq=80m +deneyapkartv2.menu.FlashMode.qio120=QIO 120MHz +deneyapkartv2.menu.FlashMode.qio120.build.flash_mode=dio +deneyapkartv2.menu.FlashMode.qio120.build.boot=qio +deneyapkartv2.menu.FlashMode.qio120.build.boot_freq=120m +deneyapkartv2.menu.FlashMode.qio120.build.flash_freq=80m +deneyapkartv2.menu.FlashMode.dio=DIO 80MHz +deneyapkartv2.menu.FlashMode.dio.build.flash_mode=dio +deneyapkartv2.menu.FlashMode.dio.build.boot=dio +deneyapkartv2.menu.FlashMode.dio.build.boot_freq=80m +deneyapkartv2.menu.FlashMode.dio.build.flash_freq=80m +deneyapkartv2.menu.FlashMode.opi=OPI 80MHz +deneyapkartv2.menu.FlashMode.opi.build.flash_mode=dout +deneyapkartv2.menu.FlashMode.opi.build.boot=opi +deneyapkartv2.menu.FlashMode.opi.build.boot_freq=80m +deneyapkartv2.menu.FlashMode.opi.build.flash_freq=80m + +deneyapkartv2.menu.FlashSize.4M=4MB (32Mb) +deneyapkartv2.menu.FlashSize.4M.build.flash_size=4MB + +deneyapkartv2.menu.LoopCore.1=Core 1 +deneyapkartv2.menu.LoopCore.1.build.loop_core=-DARDUINO_RUNNING_CORE=1 +deneyapkartv2.menu.LoopCore.0=Core 0 +deneyapkartv2.menu.LoopCore.0.build.loop_core=-DARDUINO_RUNNING_CORE=0 + +deneyapkartv2.menu.EventsCore.1=Core 1 +deneyapkartv2.menu.EventsCore.1.build.event_core=-DARDUINO_EVENT_RUNNING_CORE=1 +deneyapkartv2.menu.EventsCore.0=Core 0 +deneyapkartv2.menu.EventsCore.0.build.event_core=-DARDUINO_EVENT_RUNNING_CORE=0 + +deneyapkartv2.menu.USBMode.hwcdc=Hardware CDC and JTAG +deneyapkartv2.menu.USBMode.hwcdc.build.usb_mode=1 +deneyapkartv2.menu.USBMode.default=USB-OTG (TinyUSB) +deneyapkartv2.menu.USBMode.default.build.usb_mode=0 + +deneyapkartv2.menu.CDCOnBoot.cdc=Enabled +deneyapkartv2.menu.CDCOnBoot.cdc.build.cdc_on_boot=1 +deneyapkartv2.menu.CDCOnBoot.default=Disabled +deneyapkartv2.menu.CDCOnBoot.default.build.cdc_on_boot=0 + +deneyapkartv2.menu.MSCOnBoot.default=Disabled +deneyapkartv2.menu.MSCOnBoot.default.build.msc_on_boot=0 +deneyapkartv2.menu.MSCOnBoot.msc=Enabled (Requires USB-OTG Mode) +deneyapkartv2.menu.MSCOnBoot.msc.build.msc_on_boot=1 + +deneyapkartv2.menu.DFUOnBoot.default=Disabled +deneyapkartv2.menu.DFUOnBoot.default.build.dfu_on_boot=0 +deneyapkartv2.menu.DFUOnBoot.dfu=Enabled (Requires USB-OTG Mode) +deneyapkartv2.menu.DFUOnBoot.dfu.build.dfu_on_boot=1 + +deneyapkartv2.menu.UploadMode.default=UART0 / Hardware CDC +deneyapkartv2.menu.UploadMode.default.upload.use_1200bps_touch=false +deneyapkartv2.menu.UploadMode.default.upload.wait_for_upload_port=false +deneyapkartv2.menu.UploadMode.cdc=USB-OTG CDC (TinyUSB) +deneyapkartv2.menu.UploadMode.cdc.upload.use_1200bps_touch=true +deneyapkartv2.menu.UploadMode.cdc.upload.wait_for_upload_port=true + +deneyapkartv2.menu.PartitionScheme.default=Default 4MB with spiffs (1.2MB APP/1.5MB SPIFFS) +deneyapkartv2.menu.PartitionScheme.default.build.partitions=default +deneyapkartv2.menu.PartitionScheme.defaultffat=Default 4MB with ffat (1.2MB APP/1.5MB FATFS) +deneyapkartv2.menu.PartitionScheme.defaultffat.build.partitions=default_ffat +deneyapkartv2.menu.PartitionScheme.minimal=Minimal (1.3MB APP/700KB SPIFFS) +deneyapkartv2.menu.PartitionScheme.minimal.build.partitions=minimal +deneyapkartv2.menu.PartitionScheme.no_fs=No FS 4MB (2MB APP x2) +deneyapkartv2.menu.PartitionScheme.no_fs.build.partitions=no_fs +deneyapkartv2.menu.PartitionScheme.no_fs.upload.maximum_size=2031616 +deneyapkartv2.menu.PartitionScheme.no_ota=No OTA (2MB APP/2MB SPIFFS) +deneyapkartv2.menu.PartitionScheme.no_ota.build.partitions=no_ota +deneyapkartv2.menu.PartitionScheme.no_ota.upload.maximum_size=2097152 +deneyapkartv2.menu.PartitionScheme.noota_3g=No OTA (1MB APP/3MB SPIFFS) +deneyapkartv2.menu.PartitionScheme.noota_3g.build.partitions=noota_3g +deneyapkartv2.menu.PartitionScheme.noota_3g.upload.maximum_size=1048576 +deneyapkartv2.menu.PartitionScheme.noota_ffat=No OTA (2MB APP/2MB FATFS) +deneyapkartv2.menu.PartitionScheme.noota_ffat.build.partitions=noota_ffat +deneyapkartv2.menu.PartitionScheme.noota_ffat.upload.maximum_size=2097152 +deneyapkartv2.menu.PartitionScheme.noota_3gffat=No OTA (1MB APP/3MB FATFS) +deneyapkartv2.menu.PartitionScheme.noota_3gffat.build.partitions=noota_3gffat +deneyapkartv2.menu.PartitionScheme.noota_3gffat.upload.maximum_size=1048576 +deneyapkartv2.menu.PartitionScheme.huge_app=Huge APP (3MB No OTA/1MB SPIFFS) +deneyapkartv2.menu.PartitionScheme.huge_app.build.partitions=huge_app +deneyapkartv2.menu.PartitionScheme.huge_app.upload.maximum_size=3145728 +deneyapkartv2.menu.PartitionScheme.min_spiffs=Minimal SPIFFS (1.9MB APP with OTA/190KB SPIFFS) +deneyapkartv2.menu.PartitionScheme.min_spiffs.build.partitions=min_spiffs +deneyapkartv2.menu.PartitionScheme.min_spiffs.upload.maximum_size=1966080 +deneyapkartv2.menu.PartitionScheme.rainmaker=RainMaker 4MB +deneyapkartv2.menu.PartitionScheme.rainmaker.build.partitions=rainmaker +deneyapkartv2.menu.PartitionScheme.rainmaker.upload.maximum_size=1966080 +deneyapkartv2.menu.PartitionScheme.rainmaker_4MB=RainMaker 4MB No OTA +deneyapkartv2.menu.PartitionScheme.rainmaker_4MB.build.partitions=rainmaker_4MB_no_ota +deneyapkartv2.menu.PartitionScheme.rainmaker_4MB.upload.maximum_size=4038656 +deneyapkartv2.menu.PartitionScheme.zigbee_zczr=Zigbee ZCZR 4MB with spiffs +deneyapkartv2.menu.PartitionScheme.zigbee_zczr.build.partitions=zigbee_zczr +deneyapkartv2.menu.PartitionScheme.zigbee_zczr.upload.maximum_size=1310720 +deneyapkartv2.menu.PartitionScheme.custom=Custom +deneyapkartv2.menu.PartitionScheme.custom.build.partitions= +deneyapkartv2.menu.PartitionScheme.custom.upload.maximum_size=16777216 + +deneyapkartv2.menu.CPUFreq.240=240MHz (WiFi) +deneyapkartv2.menu.CPUFreq.240.build.f_cpu=240000000L +deneyapkartv2.menu.CPUFreq.160=160MHz (WiFi) +deneyapkartv2.menu.CPUFreq.160.build.f_cpu=160000000L +deneyapkartv2.menu.CPUFreq.80=80MHz (WiFi) +deneyapkartv2.menu.CPUFreq.80.build.f_cpu=80000000L +deneyapkartv2.menu.CPUFreq.40=40MHz +deneyapkartv2.menu.CPUFreq.40.build.f_cpu=40000000L +deneyapkartv2.menu.CPUFreq.20=20MHz +deneyapkartv2.menu.CPUFreq.20.build.f_cpu=20000000L +deneyapkartv2.menu.CPUFreq.10=10MHz +deneyapkartv2.menu.CPUFreq.10.build.f_cpu=10000000L + +deneyapkartv2.menu.UploadSpeed.921600=921600 +deneyapkartv2.menu.UploadSpeed.921600.upload.speed=921600 +deneyapkartv2.menu.UploadSpeed.115200=115200 +deneyapkartv2.menu.UploadSpeed.115200.upload.speed=115200 +deneyapkartv2.menu.UploadSpeed.256000.windows=256000 +deneyapkartv2.menu.UploadSpeed.256000.upload.speed=256000 +deneyapkartv2.menu.UploadSpeed.230400.windows.upload.speed=256000 +deneyapkartv2.menu.UploadSpeed.230400=230400 +deneyapkartv2.menu.UploadSpeed.230400.upload.speed=230400 +deneyapkartv2.menu.UploadSpeed.460800.linux=460800 +deneyapkartv2.menu.UploadSpeed.460800.macosx=460800 +deneyapkartv2.menu.UploadSpeed.460800.upload.speed=460800 +deneyapkartv2.menu.UploadSpeed.512000.windows=512000 +deneyapkartv2.menu.UploadSpeed.512000.upload.speed=512000 + +deneyapkartv2.menu.DebugLevel.none=None +deneyapkartv2.menu.DebugLevel.none.build.code_debug=0 +deneyapkartv2.menu.DebugLevel.error=Error +deneyapkartv2.menu.DebugLevel.error.build.code_debug=1 +deneyapkartv2.menu.DebugLevel.warn=Warn +deneyapkartv2.menu.DebugLevel.warn.build.code_debug=2 +deneyapkartv2.menu.DebugLevel.info=Info +deneyapkartv2.menu.DebugLevel.info.build.code_debug=3 +deneyapkartv2.menu.DebugLevel.debug=Debug +deneyapkartv2.menu.DebugLevel.debug.build.code_debug=4 +deneyapkartv2.menu.DebugLevel.verbose=Verbose +deneyapkartv2.menu.DebugLevel.verbose.build.code_debug=5 + +deneyapkartv2.menu.EraseFlash.none=Disabled +deneyapkartv2.menu.EraseFlash.none.upload.erase_cmd= +deneyapkartv2.menu.EraseFlash.all=Enabled +deneyapkartv2.menu.EraseFlash.all.upload.erase_cmd=-e + +deneyapkartv2.menu.ZigbeeMode.default=Disabled +deneyapkartv2.menu.ZigbeeMode.default.build.zigbee_mode= +deneyapkartv2.menu.ZigbeeMode.default.build.zigbee_libs= +deneyapkartv2.menu.ZigbeeMode.zczr=Zigbee ZCZR (coordinator/router) +deneyapkartv2.menu.ZigbeeMode.zczr.build.zigbee_mode=-DZIGBEE_MODE_ZCZR +deneyapkartv2.menu.ZigbeeMode.zczr.build.zigbee_libs=-lesp_zb_api.zczr -lzboss_stack.zczr -lzboss_port.remote + +############################################################## + deneyapkart1A.name=Deneyap Kart 1A deneyapkart1A.bootloader.tool=esptool_py diff --git a/variants/deneyapkartv2/pins_arduino.h b/variants/deneyapkartv2/pins_arduino.h new file mode 100644 index 00000000000..f7eccadb13c --- /dev/null +++ b/variants/deneyapkartv2/pins_arduino.h @@ -0,0 +1,123 @@ +#ifndef Pins_Arduino_h +#define Pins_Arduino_h + +#include +#include "soc/soc_caps.h" + +#define USB_VID 0x303A +#define USB_PID 0x82EB +#define USB_MANUFACTURER "Turkish Technology Team Foundation (T3)" +#define USB_PRODUCT "DENEYAP KART v2" +#define USB_SERIAL "" // Empty string for MAC address + +static const uint8_t LED_BUILTIN = SOC_GPIO_PIN_COUNT + 46; +#define BUILTIN_LED LED_BUILTIN // backward compatibility +#define LED_BUILTIN LED_BUILTIN // allow testing #ifdef LED_BUILTIN +#define RGB_BUILTIN LED_BUILTIN +#define RGBLED LED_BUILTIN +#define RGB_BRIGHTNESS 64 + +static const uint8_t GPKEY = 0; +#define KEY_BUILTIN GPKEY +#define BUILTIN_KEY GPKEY + +static const uint8_t TX = 43; +static const uint8_t RX = 44; +#define TX1 TX +#define RX1 RX + +static const uint8_t SDA = 47; +static const uint8_t SCL = 21; + +static const uint8_t SS = 42; +static const uint8_t MOSI = 39; +static const uint8_t MISO = 40; +static const uint8_t SCK = 41; + +static const uint8_t A0 = 4; +static const uint8_t A1 = 5; +static const uint8_t A2 = 6; +static const uint8_t A3 = 7; +static const uint8_t A4 = 15; +static const uint8_t A5 = 16; +static const uint8_t A6 = 17; +static const uint8_t A7 = 18; +static const uint8_t A8 = 8; +static const uint8_t A9 = 9; +static const uint8_t A10 = 10; +static const uint8_t A11 = 11; +static const uint8_t A12 = 2; +static const uint8_t A13 = 1; +static const uint8_t A14 = 3; +static const uint8_t A15 = 12; +static const uint8_t A16 = 13; +static const uint8_t A17 = 14; + +static const uint8_t T0 = 4; +static const uint8_t T1 = 5; +static const uint8_t T2 = 6; +static const uint8_t T3 = 7; +static const uint8_t T4 = 8; +static const uint8_t T5 = 9; +static const uint8_t T6 = 10; +static const uint8_t T7 = 11; +static const uint8_t T8 = 2; +static const uint8_t T9 = 1; +static const uint8_t T10 = 3; +static const uint8_t T11 = 12; +static const uint8_t T12 = 13; +static const uint8_t T13 = 14; + +static const uint8_t D0 = 1; +static const uint8_t D1 = 2; +static const uint8_t D2 = 43; +static const uint8_t D3 = 44; +static const uint8_t D4 = 42; +static const uint8_t D5 = 41; +static const uint8_t D6 = 40; +static const uint8_t D7 = 39; +static const uint8_t D8 = 38; +static const uint8_t D9 = 48; +static const uint8_t D10 = 47; +static const uint8_t D11 = 21; +static const uint8_t D12 = 11; +static const uint8_t D13 = 10; +static const uint8_t D14 = 9; +static const uint8_t D15 = 8; +static const uint8_t D16 = 18; +static const uint8_t D17 = 17; +static const uint8_t D18 = 16; +static const uint8_t D19 = 15; +static const uint8_t D20 = 7; +static const uint8_t D21 = 6; +static const uint8_t D22 = 5; +static const uint8_t D23 = 4; +static const uint8_t D24 = 46; +static const uint8_t D25 = 0; +static const uint8_t D26 = 3; +static const uint8_t D27 = 12; +static const uint8_t D28 = 13; +static const uint8_t D29 = 14; + +static const uint8_t CAMSD = 4; +static const uint8_t CAMSC = 5; +static const uint8_t CAMD2 = 41; +static const uint8_t CAMD3 = 2; +static const uint8_t CAMD4 = 1; +static const uint8_t CAMD5 = 42; +static const uint8_t CAMD6 = 40; +static const uint8_t CAMD7 = 38; +static const uint8_t CAMD8 = 17; +static const uint8_t CAMD9 = 15; +static const uint8_t CAMPC = 39; +static const uint8_t CAMXC = 16; +static const uint8_t CAMH = 7; +static const uint8_t CAMV = 6; + +static const uint8_t SDCM = 12; +static const uint8_t SDCK = 13; +static const uint8_t SDDA = 14; + +static const uint8_t BAT = 3; + +#endif /* Pins_Arduino_h */ From aab542d658819c6ce20035558e88911e131c8aef Mon Sep 17 00:00:00 2001 From: Me No Dev Date: Thu, 3 Jul 2025 23:14:53 +0300 Subject: [PATCH 094/173] Update Issue-report.yml to add v 3.2.1 --- .github/ISSUE_TEMPLATE/Issue-report.yml | 1 + 1 file changed, 1 insertion(+) diff --git a/.github/ISSUE_TEMPLATE/Issue-report.yml b/.github/ISSUE_TEMPLATE/Issue-report.yml index 9dba5e0ca8f..6dc1b0de171 100644 --- a/.github/ISSUE_TEMPLATE/Issue-report.yml +++ b/.github/ISSUE_TEMPLATE/Issue-report.yml @@ -43,6 +43,7 @@ body: - latest stable Release (if not listed below) - latest development Release Candidate (RC-X) - latest master (checkout manually) + - v3.2.1 - v3.2.0 - v3.1.3 - v3.1.2 From 1426927c837dfa0c10835eb3c769f7000db64c26 Mon Sep 17 00:00:00 2001 From: Me No Dev Date: Fri, 4 Jul 2025 18:37:56 +0300 Subject: [PATCH 095/173] fix(matter): Fix MatterSmartButon.ino when CHIPOBLE is on --- .../Matter/examples/MatterSmartButon/MatterSmartButon.ino | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/libraries/Matter/examples/MatterSmartButon/MatterSmartButon.ino b/libraries/Matter/examples/MatterSmartButon/MatterSmartButon.ino index 29caf00004c..f4d978175cc 100644 --- a/libraries/Matter/examples/MatterSmartButon/MatterSmartButon.ino +++ b/libraries/Matter/examples/MatterSmartButon/MatterSmartButon.ino @@ -45,12 +45,12 @@ void setup() { Serial.begin(115200); +// CONFIG_ENABLE_CHIPOBLE is enabled when BLE is used to commission the Matter Network +#if !CONFIG_ENABLE_CHIPOBLE // We start by connecting to a WiFi network Serial.print("Connecting to "); Serial.println(ssid); -// CONFIG_ENABLE_CHIPOBLE is enabled when BLE is used to commission the Matter Network -#if !CONFIG_ENABLE_CHIPOBLE // Manually connect to WiFi WiFi.begin(ssid, password); // Wait for connection From d3c5a82eed1fb30403186c738dbaa6f45cac93bc Mon Sep 17 00:00:00 2001 From: "Daniel.Cao" <144674500+DanielCao0@users.noreply.github.com> Date: Mon, 7 Jul 2025 16:56:44 +0800 Subject: [PATCH 096/173] fix(board): Update PSRAM configuration for RAK3112 to fix PSRAM error (#11552) * fix(board): Update PSRAM configuration for RAK3112 to fix PSRAM error * feat(board): RAK3112 add WisBlock module pin definitions to pins_arduino.h * fix(board): Update RAK3112 flash mode and boot settings for improved performance * ci(pre-commit): Apply automatic fixes --------- Co-authored-by: Daniel.Cao Co-authored-by: pre-commit-ci-lite[bot] <117423508+pre-commit-ci-lite[bot]@users.noreply.github.com> --- boards.txt | 6 ++++-- variants/rakwireless_rak3112/pins_arduino.h | 13 +++++++++++++ 2 files changed, 17 insertions(+), 2 deletions(-) diff --git a/boards.txt b/boards.txt index e47995985ed..ac2b1a336e4 100644 --- a/boards.txt +++ b/boards.txt @@ -50805,10 +50805,12 @@ rakwireless_rak3112.build.dfu_on_boot=0 rakwireless_rak3112.build.f_cpu=240000000L rakwireless_rak3112.build.flash_size=16MB rakwireless_rak3112.build.flash_freq=80m -rakwireless_rak3112.build.flash_mode=dio -rakwireless_rak3112.build.boot=dio +rakwireless_rak3112.build.flash_mode=qio +rakwireless_rak3112.build.boot=qio rakwireless_rak3112.build.partitions=default rakwireless_rak3112.build.defines= +rakwireless_rak3112.build.psram_type=opi +rakwireless_rak3112.build.memory_type={build.boot}_{build.psram_type} rakwireless_rak3112.menu.PSRAM.enabled=Enabled rakwireless_rak3112.menu.PSRAM.enabled.build.defines=-DBOARD_HAS_PSRAM diff --git a/variants/rakwireless_rak3112/pins_arduino.h b/variants/rakwireless_rak3112/pins_arduino.h index 5d1e451494a..f1bcc7a6120 100644 --- a/variants/rakwireless_rak3112/pins_arduino.h +++ b/variants/rakwireless_rak3112/pins_arduino.h @@ -47,4 +47,17 @@ static const uint8_t SCK = 13; #define LORA_BUSY 48 #define LORA_IRQ LORA_DIO1 +// For WisBlock modules, see: https://docs.rakwireless.com/Product-Categories/WisBlock/ +#define WB_IO1 21 +#define WB_IO2 14 +#define WB_IO3 41 +#define WB_IO4 42 +#define WB_IO5 38 +#define WB_IO6 39 +#define WB_A0 1 +#define WB_A1 2 +#define WB_CS 12 +#define WB_LED1 46 +#define WB_LED2 45 + #endif /* Pins_Arduino_h */ From b709a782839e4bfb143df5be9a108ff4414a565c Mon Sep 17 00:00:00 2001 From: Jason2866 <24528715+Jason2866@users.noreply.github.com> Date: Mon, 7 Jul 2025 12:01:50 +0200 Subject: [PATCH 097/173] fix deprecated warnings caaused from esptool v5.0.0 (#11556) --- tools/pioarduino-build.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tools/pioarduino-build.py b/tools/pioarduino-build.py index b67580e264c..4d4161dd9ca 100644 --- a/tools/pioarduino-build.py +++ b/tools/pioarduino-build.py @@ -95,7 +95,7 @@ def generate_bootloader_image(bootloader_elf): env.VerboseAction( " ".join( [ - '"$PYTHONEXE" "$OBJCOPY"', + "$OBJCOPY", "--chip", build_mcu, "elf2image", From e2c7578fa87e9f20b0a86090b23d89ae22a0146f Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Jan=20Proch=C3=A1zka?= <90197375+P-R-O-C-H-Y@users.noreply.github.com> Date: Tue, 8 Jul 2025 12:08:56 +0200 Subject: [PATCH 098/173] feat(zigbee): Add Fan Control endpoint support (#11559) * feat(zigbee): Add Fan Control endpoint support * fix(zigbee): Update logs and change device_id * ci(pre-commit): Apply automatic fixes --------- Co-authored-by: pre-commit-ci-lite[bot] <117423508+pre-commit-ci-lite[bot]@users.noreply.github.com> --- CMakeLists.txt | 1 + .../examples/Zigbee_Fan_Control/README.md | 83 +++++++++++ .../Zigbee_Fan_Control/Zigbee_Fan_Control.ino | 129 ++++++++++++++++++ .../examples/Zigbee_Fan_Control/ci.json | 6 + libraries/Zigbee/keywords.txt | 60 ++++++-- libraries/Zigbee/src/Zigbee.h | 1 + libraries/Zigbee/src/ep/ZigbeeFanControl.cpp | 60 ++++++++ libraries/Zigbee/src/ep/ZigbeeFanControl.h | 65 +++++++++ 8 files changed, 393 insertions(+), 12 deletions(-) create mode 100644 libraries/Zigbee/examples/Zigbee_Fan_Control/README.md create mode 100644 libraries/Zigbee/examples/Zigbee_Fan_Control/Zigbee_Fan_Control.ino create mode 100644 libraries/Zigbee/examples/Zigbee_Fan_Control/ci.json create mode 100644 libraries/Zigbee/src/ep/ZigbeeFanControl.cpp create mode 100644 libraries/Zigbee/src/ep/ZigbeeFanControl.h diff --git a/CMakeLists.txt b/CMakeLists.txt index e8f44ac5ee0..f21183ee11e 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -305,6 +305,7 @@ set(ARDUINO_LIBRARY_Zigbee_SRCS libraries/Zigbee/src/ep/ZigbeeElectricalMeasurement.cpp libraries/Zigbee/src/ep/ZigbeeBinary.cpp libraries/Zigbee/src/ep/ZigbeePowerOutlet.cpp + libraries/Zigbee/src/ep/ZigbeeFanControl.cpp ) set(ARDUINO_LIBRARY_BLE_SRCS diff --git a/libraries/Zigbee/examples/Zigbee_Fan_Control/README.md b/libraries/Zigbee/examples/Zigbee_Fan_Control/README.md new file mode 100644 index 00000000000..91700b669a0 --- /dev/null +++ b/libraries/Zigbee/examples/Zigbee_Fan_Control/README.md @@ -0,0 +1,83 @@ +# Arduino-ESP32 Zigbee Fan Control Example + +This example demonstrates how to use the Zigbee library to create a router device fan control and use it as a Home Automation (HA) fan control device. + +# Supported Targets + +Currently, this example supports the following targets. + +| Supported Targets | ESP32-C6 | ESP32-H2 | +| ----------------- | -------- | -------- | + +## Fan Control Functions + +1. Initialize a Zigbee fan control device. +2. Control fan modes (OFF, LOW, MEDIUM, HIGH, ON). +3. Respond to fan control commands from the Zigbee network. + +## Hardware Required + +* ESP32-H2 or ESP32-C6 development board +* A USB cable for power supply and programming +* RGB LED for visual feedback (built-in on most development boards) + +### Configure the Project + +In this example the RGB LED is used to indicate the current fan control mode. +The LED colors represent different fan modes: +- OFF: No light +- LOW: Blue +- MEDIUM: Yellow +- HIGH: Red +- ON: White + +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 ZCZR (coordinator/router)` +* Select Partition Scheme for Zigbee: `Tools -> Partition Scheme: Zigbee ZCZR 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_Fan_Control/Zigbee_Fan_Control.ino b/libraries/Zigbee/examples/Zigbee_Fan_Control/Zigbee_Fan_Control.ino new file mode 100644 index 00000000000..4c0d15aa563 --- /dev/null +++ b/libraries/Zigbee/examples/Zigbee_Fan_Control/Zigbee_Fan_Control.ino @@ -0,0 +1,129 @@ +// Copyright 2025 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 simple Zigbee fan control. + * + * The example demonstrates how to use Zigbee library to create a router device fan control. + * The fan control is a Zigbee router 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 Jan Procházka (https://github.com/P-R-O-C-H-Y/) + */ + +#ifndef ZIGBEE_MODE_ZCZR +#error "Zigbee coordinator mode is not selected in Tools->Zigbee mode" +#endif + +#include "Zigbee.h" + +/* Zigbee light bulb configuration */ +#define ZIGBEE_FAN_CONTROL_ENDPOINT 1 + +#ifdef RGB_BUILTIN +uint8_t led = RGB_BUILTIN; // To demonstrate the current fan control mode +#else +uint8_t led = 2; +#endif + +uint8_t button = BOOT_PIN; + +ZigbeeFanControl zbFanControl = ZigbeeFanControl(ZIGBEE_FAN_CONTROL_ENDPOINT); + +/********************* fan control callback function **************************/ +void setFan(ZigbeeFanMode mode) { + switch (mode) { + case FAN_MODE_OFF: + rgbLedWrite(led, 0, 0, 0); // Off + Serial.println("Fan mode: OFF"); + break; + case FAN_MODE_LOW: + rgbLedWrite(led, 0, 0, 255); // Blue + Serial.println("Fan mode: LOW"); + break; + case FAN_MODE_MEDIUM: + rgbLedWrite(led, 255, 255, 0); // Yellow + Serial.println("Fan mode: MEDIUM"); + break; + case FAN_MODE_HIGH: + rgbLedWrite(led, 255, 0, 0); // Red + Serial.println("Fan mode: HIGH"); + break; + case FAN_MODE_ON: + rgbLedWrite(led, 255, 255, 255); // White + Serial.println("Fan mode: ON"); + break; + default: log_e("Unhandled fan mode: %d", mode); break; + } +} + +/********************* Arduino functions **************************/ +void setup() { + Serial.begin(115200); + + // Init LED that will be used to indicate the current fan control mode + rgbLedWrite(led, 0, 0, 0); + + // Init button for factory reset + pinMode(button, INPUT_PULLUP); + + //Optional: set Zigbee device name and model + zbFanControl.setManufacturerAndModel("Espressif", "ZBFanControl"); + + // Set the fan mode sequence to LOW_MED_HIGH + zbFanControl.setFanModeSequence(FAN_MODE_SEQUENCE_LOW_MED_HIGH); + + // Set callback function for fan mode change + zbFanControl.onFanModeChange(setFan); + + //Add endpoint to Zigbee Core + Serial.println("Adding ZigbeeFanControl endpoint to Zigbee Core"); + Zigbee.addEndpoint(&zbFanControl); + + // When all EPs are registered, start Zigbee in ROUTER mode + if (!Zigbee.begin(ZIGBEE_ROUTER)) { + Serial.println("Zigbee failed to start!"); + Serial.println("Rebooting..."); + ESP.restart(); + } + Serial.println("Connecting to network"); + while (!Zigbee.connected()) { + Serial.print("."); + delay(100); + } + Serial.println(); +} + +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(); + } + } + } + delay(100); +} diff --git a/libraries/Zigbee/examples/Zigbee_Fan_Control/ci.json b/libraries/Zigbee/examples/Zigbee_Fan_Control/ci.json new file mode 100644 index 00000000000..15d6190e4ae --- /dev/null +++ b/libraries/Zigbee/examples/Zigbee_Fan_Control/ci.json @@ -0,0 +1,6 @@ +{ + "fqbn_append": "PartitionScheme=zigbee_zczr,ZigbeeMode=zczr", + "requires": [ + "CONFIG_ZB_ENABLED=y" + ] +} diff --git a/libraries/Zigbee/keywords.txt b/libraries/Zigbee/keywords.txt index 556b6408ea2..23f3af3bf02 100644 --- a/libraries/Zigbee/keywords.txt +++ b/libraries/Zigbee/keywords.txt @@ -12,25 +12,31 @@ Zigbee KEYWORD1 ZigbeeEP KEYWORD1 # Endpoint Classes -ZigbeeLight KEYWORD1 -ZigbeeSwitch KEYWORD1 -ZigbeeColorDimmableLight KEYWORD1 -ZigbeeColorDimmerSwitch KEYWORD1 -ZigbeeTempSensor KEYWORD1 -ZigbeeThermostat KEYWORD1 -ZigbeeFlowSensor KEYWORD1 -ZigbeePressureSensor KEYWORD1 -ZigbeeOccupancySensor KEYWORD1 ZigbeeAnalog KEYWORD1 +ZigbeeBinary KEYWORD1 ZigbeeCarbonDioxideSensor KEYWORD1 +ZigbeeColorDimmableLight KEYWORD1 +ZigbeeColorDimmerSwitch KEYWORD1 ZigbeeContactSwitch KEYWORD1 +ZigbeeDimableLight KEYWORD1 ZigbeeDoorWindowHandle KEYWORD1 +ZigbeeElectricalMeasurement KEYWORD1 +ZigbeeFanControl KEYWORD1 +ZigbeeFlowSensor KEYWORD1 ZigbeeGateway KEYWORD1 +ZigbeeIlluminanceSensor KEYWORD1 +ZigbeeLight KEYWORD1 +ZigbeeOccupancySensor KEYWORD1 +ZigbeePM25Sensor KEYWORD1 +ZigbeePowerOutlet KEYWORD1 +ZigbeePressureSensor KEYWORD1 ZigbeeRangeExtender KEYWORD1 +ZigbeeSwitch KEYWORD1 +ZigbeeTempSensor KEYWORD1 +ZigbeeThermostat KEYWORD1 ZigbeeVibrationSensor KEYWORD1 ZigbeeWindowCovering KEYWORD1 -ZigbeeIlluminanceSensor KEYWORD1 -ZigbeePowerOutlet KEYWORD1 +ZigbeeWindSpeedSensor KEYWORD1 # Other zigbee_role_t KEYWORD1 @@ -39,6 +45,8 @@ zb_device_params_t KEYWORD1 zigbee_scan_result_t KEYWORD1 zb_power_source_t KEYWORD1 ZigbeeWindowCoveringType KEYWORD1 +ZigbeeFanMode KEYWORD1 +ZigbeeFanModeSequence KEYWORD1 ####################################### # Methods and Functions (KEYWORD2) @@ -73,6 +81,7 @@ printBoundDevices KEYWORD2 getBoundDevices KEYWORD2 bound KEYWORD2 allowMultipleBinding KEYWORD2 +setManualBinding KEYWORD2 setManufacturerAndModel KEYWORD2 setPowerSource KEYWORD2 setBatteryPercentage KEYWORD2 @@ -80,6 +89,13 @@ reportBatteryPercentage KEYWORD2 readManufacturer KEYWORD2 readModel KEYWORD2 onIdentify KEYWORD2 +addTimeCluster KEYWORD2 +setTime KEYWORD2 +setTimezone KEYWORD2 +getTime KEYWORD2 +getTimezone KEYWORD2 +addOTAClient KEYWORD2 +clearBoundDevices KEYWORD2 # ZigbeeLight + ZigbeeColorDimmableLight onLightChange KEYWORD2 @@ -171,7 +187,7 @@ setTilted KEYWORD2 # ZigbeeVibrationSensor setVibration KEYWORD2 -ZigbeeWindowCovering +# ZigbeeWindowCovering onOpen KEYWORD2 onClose KEYWORD2 onGoToLiftPercentage KEYWORD2 @@ -186,6 +202,26 @@ setConfigStatus KEYWORD2 setMode KEYWORD2 setLimits KEYWORD2 +# ZigbeeBinary +addBinaryInput KEYWORD2 +addBinaryOutput KEYWORD2 +onBinaryOutputChange KEYWORD2 +setBinaryInput KEYWORD2 +setBinaryOutput KEYWORD2 +getBinaryOutput KEYWORD2 +reportBinaryInput KEYWORD2 +reportBinaryOutput KEYWORD2 +setBinaryInputApplication KEYWORD2 +setBinaryInputDescription KEYWORD2 +setBinaryOutputApplication KEYWORD2 +setBinaryOutputDescription KEYWORD2 + +# ZigbeeFanControl +setFanModeSequence KEYWORD2 +getFanMode KEYWORD2 +getFanModeSequence KEYWORD2 +onFanModeChange KEYWORD2 + ####################################### # Constants (LITERAL1) ####################################### diff --git a/libraries/Zigbee/src/Zigbee.h b/libraries/Zigbee/src/Zigbee.h index b2e2e5dd027..65c9e7f0daa 100644 --- a/libraries/Zigbee/src/Zigbee.h +++ b/libraries/Zigbee/src/Zigbee.h @@ -16,6 +16,7 @@ #include "ep/ZigbeeLight.h" //// Controllers #include "ep/ZigbeeThermostat.h" +#include "ep/ZigbeeFanControl.h" ////Outlets #include "ep/ZigbeePowerOutlet.h" //// Sensors diff --git a/libraries/Zigbee/src/ep/ZigbeeFanControl.cpp b/libraries/Zigbee/src/ep/ZigbeeFanControl.cpp new file mode 100644 index 00000000000..f4b32ce1200 --- /dev/null +++ b/libraries/Zigbee/src/ep/ZigbeeFanControl.cpp @@ -0,0 +1,60 @@ +#include "ZigbeeFanControl.h" +#if CONFIG_ZB_ENABLED + +ZigbeeFanControl::ZigbeeFanControl(uint8_t endpoint) : ZigbeeEP(endpoint) { + _device_id = ESP_ZB_HA_THERMOSTAT_DEVICE_ID; //There is no FAN_CONTROL_DEVICE_ID in the Zigbee spec + + //Create basic analog sensor clusters without configuration + _cluster_list = esp_zb_zcl_cluster_list_create(); + esp_zb_cluster_list_add_basic_cluster(_cluster_list, esp_zb_basic_cluster_create(NULL), ESP_ZB_ZCL_CLUSTER_SERVER_ROLE); + esp_zb_cluster_list_add_identify_cluster(_cluster_list, esp_zb_identify_cluster_create(NULL), ESP_ZB_ZCL_CLUSTER_SERVER_ROLE); + esp_zb_cluster_list_add_fan_control_cluster(_cluster_list, esp_zb_fan_control_cluster_create(NULL), ESP_ZB_ZCL_CLUSTER_SERVER_ROLE); + + _ep_config = { + .endpoint = _endpoint, .app_profile_id = ESP_ZB_AF_HA_PROFILE_ID, .app_device_id = ESP_ZB_HA_HEATING_COOLING_UNIT_DEVICE_ID, .app_device_version = 0 + }; +} + +bool ZigbeeFanControl::setFanModeSequence(ZigbeeFanModeSequence sequence) { + esp_zb_attribute_list_t *fan_control_cluster = + esp_zb_cluster_list_get_cluster(_cluster_list, ESP_ZB_ZCL_CLUSTER_ID_FAN_CONTROL, ESP_ZB_ZCL_CLUSTER_SERVER_ROLE); + esp_err_t ret = esp_zb_cluster_update_attr(fan_control_cluster, ESP_ZB_ZCL_ATTR_FAN_CONTROL_FAN_MODE_SEQUENCE_ID, (void *)&sequence); + if (ret != ESP_OK) { + log_e("Failed to set min value: 0x%x: %s", ret, esp_err_to_name(ret)); + return false; + } + _current_fan_mode_sequence = sequence; + _current_fan_mode = FAN_MODE_OFF; + // Set initial fan mode to OFF + ret = esp_zb_cluster_update_attr(fan_control_cluster, ESP_ZB_ZCL_ATTR_FAN_CONTROL_FAN_MODE_ID, (void *)&_current_fan_mode); + if (ret != ESP_OK) { + log_e("Failed to set fan mode: 0x%x: %s", ret, esp_err_to_name(ret)); + return false; + } + return true; +} + +//set attribute method -> method overridden in child class +void ZigbeeFanControl::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_FAN_CONTROL) { + if (message->attribute.id == ESP_ZB_ZCL_ATTR_FAN_CONTROL_FAN_MODE_ID && message->attribute.data.type == ESP_ZB_ZCL_ATTR_TYPE_8BIT_ENUM) { + _current_fan_mode = *(ZigbeeFanMode *)message->attribute.data.value; + fanModeChanged(); + } else { + log_w("Received message ignored. Attribute ID: %d not supported for Fan Control", message->attribute.id); + } + } else { + log_w("Received message ignored. Cluster ID: %d not supported for Fan Control", message->info.cluster); + } +} + +void ZigbeeFanControl::fanModeChanged() { + if (_on_fan_mode_change) { + _on_fan_mode_change(_current_fan_mode); + } else { + log_w("No callback function set for fan mode change"); + } +} + +#endif // CONFIG_ZB_ENABLED diff --git a/libraries/Zigbee/src/ep/ZigbeeFanControl.h b/libraries/Zigbee/src/ep/ZigbeeFanControl.h new file mode 100644 index 00000000000..25b5862c5c4 --- /dev/null +++ b/libraries/Zigbee/src/ep/ZigbeeFanControl.h @@ -0,0 +1,65 @@ +/* Class of Zigbee Pressure sensor endpoint inherited from common EP class */ + +#pragma once + +#include "soc/soc_caps.h" +#include "sdkconfig.h" +#if CONFIG_ZB_ENABLED + +#include "ZigbeeEP.h" +#include "ha/esp_zigbee_ha_standard.h" + +// Custom Arduino-friendly enums for fan mode values +enum ZigbeeFanMode { + FAN_MODE_OFF = ESP_ZB_ZCL_FAN_CONTROL_FAN_MODE_OFF, + FAN_MODE_LOW = ESP_ZB_ZCL_FAN_CONTROL_FAN_MODE_LOW, + FAN_MODE_MEDIUM = ESP_ZB_ZCL_FAN_CONTROL_FAN_MODE_MEDIUM, + FAN_MODE_HIGH = ESP_ZB_ZCL_FAN_CONTROL_FAN_MODE_HIGH, + FAN_MODE_ON = ESP_ZB_ZCL_FAN_CONTROL_FAN_MODE_ON, + FAN_MODE_AUTO = ESP_ZB_ZCL_FAN_CONTROL_FAN_MODE_AUTO, + FAN_MODE_SMART = ESP_ZB_ZCL_FAN_CONTROL_FAN_MODE_SMART, +}; + +// Custom Arduino-friendly enums for fan mode sequence +enum ZigbeeFanModeSequence { + FAN_MODE_SEQUENCE_LOW_MED_HIGH = ESP_ZB_ZCL_FAN_CONTROL_FAN_MODE_SEQUENCE_LOW_MED_HIGH, + FAN_MODE_SEQUENCE_LOW_HIGH = ESP_ZB_ZCL_FAN_CONTROL_FAN_MODE_SEQUENCE_LOW_HIGH, + FAN_MODE_SEQUENCE_LOW_MED_HIGH_AUTO = ESP_ZB_ZCL_FAN_CONTROL_FAN_MODE_SEQUENCE_LOW_MED_HIGH_AUTO, + FAN_MODE_SEQUENCE_LOW_HIGH_AUTO = ESP_ZB_ZCL_FAN_CONTROL_FAN_MODE_SEQUENCE_LOW_HIGH_AUTO, + FAN_MODE_SEQUENCE_ON_AUTO = ESP_ZB_ZCL_FAN_CONTROL_FAN_MODE_SEQUENCE_ON_AUTO, +}; + +class ZigbeeFanControl : public ZigbeeEP { +public: + ZigbeeFanControl(uint8_t endpoint); + ~ZigbeeFanControl() {} + + // Set the fan mode sequence value + bool setFanModeSequence(ZigbeeFanModeSequence sequence); + + // Use to get fan mode + ZigbeeFanMode getFanMode() { + return _current_fan_mode; + } + + // Use to get fan mode sequence + ZigbeeFanModeSequence getFanModeSequence() { + return _current_fan_mode_sequence; + } + + // On fan mode change callback + void onFanModeChange(void (*callback)(ZigbeeFanMode mode)) { + _on_fan_mode_change = callback; + } + +private: + void zbAttributeSet(const esp_zb_zcl_set_attr_value_message_t *message) override; + //callback function to be called on fan mode change + void (*_on_fan_mode_change)(ZigbeeFanMode mode); + void fanModeChanged(); + + ZigbeeFanMode _current_fan_mode; + ZigbeeFanModeSequence _current_fan_mode_sequence; +}; + +#endif // CONFIG_ZB_ENABLED From 040e0ca42a04e644e5bffee62e2ce211aa3e02be Mon Sep 17 00:00:00 2001 From: Lucas Saavedra Vaz <32426024+lucasssvaz@users.noreply.github.com> Date: Tue, 8 Jul 2025 07:09:59 -0300 Subject: [PATCH 099/173] fix(dangerjs): Disable target branch rule (#11565) --- .github/workflows/dangerjs.yml | 1 + 1 file changed, 1 insertion(+) diff --git a/.github/workflows/dangerjs.yml b/.github/workflows/dangerjs.yml index 13bc907566b..bba96bfedee 100644 --- a/.github/workflows/dangerjs.yml +++ b/.github/workflows/dangerjs.yml @@ -24,4 +24,5 @@ jobs: instructions-cla-link: "https://cla-assistant.io/espressif/arduino-esp32" instructions-contributions-file: "docs/en/contributing.rst" rule-max-commits: "false" + rule-target-branch: "false" commit-messages-min-summary-length: "10" From 241e2576be918218e10a615de851f61b8ebd5936 Mon Sep 17 00:00:00 2001 From: Me No Dev Date: Tue, 8 Jul 2025 16:00:51 +0300 Subject: [PATCH 100/173] fix(async): Update IP setup in AsyncUDP (#11569) * fix(async): Update IP setup in AsyncUDP * fix(udp): Revert to IP_SET_TYPE_VAL in connect --- libraries/AsyncUDP/src/AsyncUDP.cpp | 17 ++++------------- 1 file changed, 4 insertions(+), 13 deletions(-) diff --git a/libraries/AsyncUDP/src/AsyncUDP.cpp b/libraries/AsyncUDP/src/AsyncUDP.cpp index cab9c951921..2d533831cd5 100644 --- a/libraries/AsyncUDP/src/AsyncUDP.cpp +++ b/libraries/AsyncUDP/src/AsyncUDP.cpp @@ -582,8 +582,8 @@ bool AsyncUDP::listen(const ip_addr_t *addr, uint16_t port) { } close(); if (addr) { - IP_SET_TYPE_VAL(_pcb->local_ip, addr->type); - IP_SET_TYPE_VAL(_pcb->remote_ip, addr->type); + IP_SET_TYPE_VAL(_pcb->local_ip, IP_GET_TYPE(addr)); + IP_SET_TYPE_VAL(_pcb->remote_ip, IP_GET_TYPE(addr)); } if (_udp_bind(_pcb, addr, port) != ERR_OK) { return false; @@ -692,17 +692,8 @@ bool AsyncUDP::listenMulticast(const ip_addr_t *addr, uint16_t port, uint8_t ttl return false; } -#if CONFIG_LWIP_IPV6 - if (IP_IS_V6(addr)) { - IP_SET_TYPE(&bind_addr, IPADDR_TYPE_V6); - ip6_addr_set_any(&bind_addr.u_addr.ip6); - } else { -#endif - IP_SET_TYPE(&bind_addr, IPADDR_TYPE_V4); - ip4_addr_set_any(&bind_addr.u_addr.ip4); -#if CONFIG_LWIP_IPV6 - } -#endif + IP_SET_TYPE(&bind_addr, IP_GET_TYPE(addr)); + ip_addr_set_any(IP_IS_V6(addr), &bind_addr); if (!listen(&bind_addr, port)) { return false; } From 2cb6fbccdbb6f07e1d20951727875613047a235f Mon Sep 17 00:00:00 2001 From: Me No Dev Date: Tue, 8 Jul 2025 16:18:42 +0300 Subject: [PATCH 101/173] Add access methods to get the Wire bus number and I2C bus handle (#11570) * feat(i2c): Add method to access the I2C bus handle * feat(wire): Add access method to get the I2C bus number * ci(pre-commit): Apply automatic fixes --------- Co-authored-by: pre-commit-ci-lite[bot] <117423508+pre-commit-ci-lite[bot]@users.noreply.github.com> --- cores/esp32/esp32-hal-i2c-ng.c | 7 +++++++ cores/esp32/esp32-hal-i2c.h | 5 +++++ libraries/Wire/src/Wire.cpp | 6 +++++- libraries/Wire/src/Wire.h | 2 ++ 4 files changed, 19 insertions(+), 1 deletion(-) diff --git a/cores/esp32/esp32-hal-i2c-ng.c b/cores/esp32/esp32-hal-i2c-ng.c index 8e48d0e0397..a3b2307b8a8 100644 --- a/cores/esp32/esp32-hal-i2c-ng.c +++ b/cores/esp32/esp32-hal-i2c-ng.c @@ -56,6 +56,13 @@ static bool i2cDetachBus(void *bus_i2c_num) { return true; } +void *i2cBusHandle(uint8_t i2c_num) { + if (i2c_num >= SOC_I2C_NUM) { + return NULL; + } + return bus[i2c_num].bus_handle; +} + bool i2cIsInit(uint8_t i2c_num) { if (i2c_num >= SOC_I2C_NUM) { return false; diff --git a/cores/esp32/esp32-hal-i2c.h b/cores/esp32/esp32-hal-i2c.h index 35783d350b0..0e4f484bb46 100644 --- a/cores/esp32/esp32-hal-i2c.h +++ b/cores/esp32/esp32-hal-i2c.h @@ -19,6 +19,7 @@ #include "soc/soc_caps.h" #if SOC_I2C_SUPPORTED +#include "esp_idf_version.h" #ifdef __cplusplus extern "C" { @@ -39,6 +40,10 @@ esp_err_t i2cWriteReadNonStop( ); bool i2cIsInit(uint8_t i2c_num); +#if ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 4, 0) +void *i2cBusHandle(uint8_t i2c_num); +#endif + #ifdef __cplusplus } #endif diff --git a/libraries/Wire/src/Wire.cpp b/libraries/Wire/src/Wire.cpp index f8d9496389f..34c814b5117 100644 --- a/libraries/Wire/src/Wire.cpp +++ b/libraries/Wire/src/Wire.cpp @@ -39,7 +39,7 @@ extern "C" { #include "Arduino.h" TwoWire::TwoWire(uint8_t bus_num) - : num(bus_num & 1), sda(-1), scl(-1), bufferSize(I2C_BUFFER_LENGTH) // default Wire Buffer Size + : num(bus_num), sda(-1), scl(-1), bufferSize(I2C_BUFFER_LENGTH) // default Wire Buffer Size , rxBuffer(NULL), rxIndex(0), rxLength(0), txBuffer(NULL), txLength(0), txAddress(0), _timeOutMillis(50), nonStop(false) #if !CONFIG_DISABLE_HAL_LOCKS @@ -62,6 +62,10 @@ TwoWire::~TwoWire() { #endif } +uint8_t TwoWire::getBusNum() { + return num; +} + bool TwoWire::initPins(int sdaPin, int sclPin) { if (sdaPin < 0) { // default param passed if (num == 0) { diff --git a/libraries/Wire/src/Wire.h b/libraries/Wire/src/Wire.h index 0deab7d4a57..b84aa5b2131 100644 --- a/libraries/Wire/src/Wire.h +++ b/libraries/Wire/src/Wire.h @@ -105,6 +105,8 @@ class TwoWire : public HardwareI2C { bool end() override; + uint8_t getBusNum(); + bool setClock(uint32_t freq) override; void beginTransmission(uint8_t address) override; From 6a5839acb2ead9b98bf0761c8bef16cbf327e6ca Mon Sep 17 00:00:00 2001 From: Lucas Saavedra Vaz <32426024+lucasssvaz@users.noreply.github.com> Date: Tue, 8 Jul 2025 11:56:23 -0300 Subject: [PATCH 102/173] change(esptool): Upgrade esptool to release v5.0.0 (#11562) --- .github/scripts/package_esptool.sh | 129 ------------ .github/scripts/update_esptool.py | 236 ++++++++++++++++++++++ package/package_esp32_index.template.json | 60 +++--- 3 files changed, 266 insertions(+), 159 deletions(-) delete mode 100755 .github/scripts/package_esptool.sh create mode 100644 .github/scripts/update_esptool.py diff --git a/.github/scripts/package_esptool.sh b/.github/scripts/package_esptool.sh deleted file mode 100755 index 32b87b277e9..00000000000 --- a/.github/scripts/package_esptool.sh +++ /dev/null @@ -1,129 +0,0 @@ -#!/bin/bash - -set -euo pipefail - -# Check version argument -if [[ $# -ne 3 ]]; then - echo "Usage: $0 " - echo "Example: $0 5.0.dev1 /tmp/esptool /tmp/esptool-5.0.dev1.json" - exit 1 -fi - -VERSION=$1 -BASE_FOLDER=$2 -JSON_PATH=$3 - -export COPYFILE_DISABLE=1 - -shopt -s nullglob # So for loop doesn't run if no matches - -# Function to update JSON for a given host -function update_json_for_host { - local host=$1 - local archive=$2 - - # Extract the old url from the JSON for this host, then replace only the filename - old_url=$(jq -r --arg host "$host" ' - .packages[].tools[] | select(.name == "esptool_py") | .systems[] | select(.host == $host) | .url // empty - ' "$tmp_json") - if [[ -n "$old_url" ]]; then - base_url="${old_url%/*}" - url="$base_url/$archive" - else - echo "No old url found for $host" - exit 1 - fi - - archiveFileName="$archive" - checksum="SHA-256:$(shasum -a 256 "$archive" | awk '{print $1}')" - size=$(stat -f%z "$archive") - - # Use jq to update the JSON - jq --arg host "$host" \ - --arg url "$url" \ - --arg archiveFileName "$archiveFileName" \ - --arg checksum "$checksum" \ - --arg size "$size" \ - ' - .packages[].tools[] - |= if .name == "esptool_py" then - .systems = ( - ((.systems // []) | map(select(.host != $host))) + [{ - host: $host, - url: $url, - archiveFileName: $archiveFileName, - checksum: $checksum, - size: $size - }] - ) - else - . - end - ' "$tmp_json" > "$tmp_json.new" && mv "$tmp_json.new" "$tmp_json" -} - -cd "$BASE_FOLDER" - -# Delete all archives before starting -rm -f esptool-*.tar.gz esptool-*.zip - -for dir in esptool-*; do - # Check if directory exists and is a directory - if [[ ! -d "$dir" ]]; then - continue - fi - - base="${dir#esptool-}" - - # Add 'linux-' prefix if base doesn't contain linux/macos/win64 - if [[ "$base" != *linux* && "$base" != *macos* && "$base" != *win64* ]]; then - base="linux-${base}" - fi - - if [[ "$dir" == esptool-win* ]]; then - # Windows zip archive - zipfile="esptool-v${VERSION}-${base}.zip" - echo "Creating $zipfile from $dir ..." - zip -r "$zipfile" "$dir" - else - # Non-Windows: set permissions and tar.gz archive - tarfile="esptool-v${VERSION}-${base}.tar.gz" - echo "Setting permissions and creating $tarfile from $dir ..." - chmod -R u=rwx,g=rx,o=rx "$dir" - tar -cvzf "$tarfile" "$dir" - fi -done - -# After the for loop, update the JSON for each archive -# Create a temporary JSON file to accumulate changes -tmp_json="${JSON_PATH}.tmp" -cp "$JSON_PATH" "$tmp_json" - -for archive in esptool-v"${VERSION}"-*.tar.gz esptool-v"${VERSION}"-*.zip; do - [ -f "$archive" ] || continue - - echo "Updating JSON for $archive" - - # Determine host from archive name - case "$archive" in - *linux-amd64*) host="x86_64-pc-linux-gnu" ;; - *linux-armv7*) host="arm-linux-gnueabihf" ;; - *linux-aarch64*) host="aarch64-linux-gnu" ;; - *macos-amd64*) host="x86_64-apple-darwin" ;; - *macos-arm64*) host="arm64-apple-darwin" ;; - *win64*) hosts=("x86_64-mingw32" "i686-mingw32") ;; - *) echo "Unknown host for $archive"; continue ;; - esac - - # For win64, loop over both hosts; otherwise, use a single host - if [[ "$archive" == *win64* ]]; then - for host in "${hosts[@]}"; do - update_json_for_host "$host" "$archive" - done - else - update_json_for_host "$host" "$archive" - fi -done - -# After all archives are processed, move the temporary JSON to the final file -mv "$tmp_json" "$JSON_PATH" diff --git a/.github/scripts/update_esptool.py b/.github/scripts/update_esptool.py new file mode 100644 index 00000000000..d99462fcb8f --- /dev/null +++ b/.github/scripts/update_esptool.py @@ -0,0 +1,236 @@ +#!/usr/bin/env python3 + +# This script is used to re-package the esptool if needed and update the JSON file +# for the Arduino ESP32 platform. +# +# The script has only been tested on macOS. +# +# For regular esptool releases, the generated packages already contain the correct permissions, +# extensions and are uploaded to the GitHub release assets. In this case, the script will only +# update the JSON file with the information from the GitHub release. +# +# The script can be used in two modes: +# 1. Local build: The build artifacts must be already downloaded and extracted in the base_folder. +# This is useful for esptool versions that are not yet released and that are grabbed from the +# GitHub build artifacts. +# 2. Release build: The script will get the release information from GitHub and update the JSON file. +# This is useful for esptool versions that are already released and that are uploaded to the +# GitHub release assets. +# +# For local build, the artifacts must be already downloaded and extracted in the base_folder +# set with the -l option. +# For example, a base folder "esptool" should contain the following folders extracted directly +# from the GitHub build artifacts: +# esptool/esptool-linux-aarch64 +# esptool/esptool-linux-amd64 +# esptool/esptool-linux-armv7 +# esptool/esptool-macos-amd64 +# esptool/esptool-macos-arm64 +# esptool/esptool-windows-amd64 + +import argparse +import json +import os +import shutil +import stat +import tarfile +import zipfile +import hashlib +import requests +from pathlib import Path + +def compute_sha256(filepath): + sha256 = hashlib.sha256() + with open(filepath, "rb") as f: + for block in iter(lambda: f.read(4096), b""): + sha256.update(block) + return f"SHA-256:{sha256.hexdigest()}" + +def get_file_size(filepath): + return os.path.getsize(filepath) + +def update_json_for_host(tmp_json_path, version, host, url, archiveFileName, checksum, size): + with open(tmp_json_path) as f: + data = json.load(f) + + for pkg in data.get("packages", []): + for tool in pkg.get("tools", []): + if tool.get("name") == "esptool_py": + tool["version"] = version + + if url is None: + # If the URL is not set, we need to find the old URL and update it + for system in tool.get("systems", []): + if system.get("host") == host: + url = system.get("url").replace(system.get("archiveFileName"), archiveFileName) + break + else: + print(f"No old URL found for host {host}. Using empty URL.") + url = "" + + # Preserve existing systems order and update or append the new system + systems = tool.get("systems", []) + system_updated = False + for i, system in enumerate(systems): + if system.get("host") == host: + systems[i] = { + "host": host, + "url": url, + "archiveFileName": archiveFileName, + "checksum": checksum, + "size": str(size), + } + system_updated = True + break + + if not system_updated: + systems.append({ + "host": host, + "url": url, + "archiveFileName": archiveFileName, + "checksum": checksum, + "size": str(size), + }) + tool["systems"] = systems + + with open(tmp_json_path, "w") as f: + json.dump(data, f, indent=2, sort_keys=False, ensure_ascii=False) + f.write("\n") + +def update_tools_dependencies(tmp_json_path, version): + with open(tmp_json_path) as f: + data = json.load(f) + + for pkg in data.get("packages", []): + for platform in pkg.get("platforms", []): + for dep in platform.get("toolsDependencies", []): + if dep.get("name") == "esptool_py": + dep["version"] = version + + with open(tmp_json_path, "w") as f: + json.dump(data, f, indent=2, sort_keys=False, ensure_ascii=False) + f.write("\n") + +def create_archives(version, base_folder): + archive_files = [] + + for dirpath in Path(base_folder).glob("esptool-*"): + if not dirpath.is_dir(): + continue + + base = dirpath.name[len("esptool-"):] + + if "windows" in dirpath.name: + zipfile_name = f"esptool-v{version}-{base}.zip" + print(f"Creating {zipfile_name} from {dirpath} ...") + with zipfile.ZipFile(zipfile_name, "w", zipfile.ZIP_DEFLATED) as zipf: + for root, _, files in os.walk(dirpath): + for file in files: + full_path = os.path.join(root, file) + zipf.write(full_path, os.path.relpath(full_path, start=dirpath)) + archive_files.append(zipfile_name) + else: + tarfile_name = f"esptool-v{version}-{base}.tar.gz" + print(f"Creating {tarfile_name} from {dirpath} ...") + for root, dirs, files in os.walk(dirpath): + for name in dirs + files: + os.chmod(os.path.join(root, name), stat.S_IRUSR | stat.S_IWUSR | stat.S_IXUSR | + stat.S_IRGRP | stat.S_IXGRP | + stat.S_IROTH | stat.S_IXOTH) + with tarfile.open(tarfile_name, "w:gz") as tar: + tar.add(dirpath, arcname=dirpath.name) + archive_files.append(tarfile_name) + + return archive_files + +def determine_hosts(archive_name): + if "linux-amd64" in archive_name: + return ["x86_64-pc-linux-gnu"] + elif "linux-armv7" in archive_name: + return ["arm-linux-gnueabihf"] + elif "linux-aarch64" in archive_name: + return ["aarch64-linux-gnu"] + elif "macos-amd64" in archive_name: + return ["x86_64-apple-darwin"] + elif "macos-arm64" in archive_name: + return ["arm64-apple-darwin"] + elif "windows-amd64" in archive_name: + return ["x86_64-mingw32", "i686-mingw32"] + else: + return [] + +def update_json_from_local_build(tmp_json_path, version, base_folder, archive_files): + for archive in archive_files: + print(f"Processing archive: {archive}") + hosts = determine_hosts(archive) + if not hosts: + print(f"Skipping unknown archive type: {archive}") + continue + + archive_path = Path(archive) + checksum = compute_sha256(archive_path) + size = get_file_size(archive_path) + + for host in hosts: + update_json_for_host(tmp_json_path, version, host, None, archive_path.name, checksum, size) + +def update_json_from_release(tmp_json_path, version, release_info): + assets = release_info.get("assets", []) + for asset in assets: + if (asset.get("name").endswith(".tar.gz") or asset.get("name").endswith(".zip")) and "esptool" in asset.get("name"): + asset_fname = asset.get("name") + print(f"Processing asset: {asset_fname}") + hosts = determine_hosts(asset_fname) + if not hosts: + print(f"Skipping unknown archive type: {asset_fname}") + continue + + asset_url = asset.get("browser_download_url") + asset_checksum = asset.get("digest").replace("sha256:", "SHA-256:") + asset_size = asset.get("size") + if asset_checksum is None: + asset_checksum = "" + print(f"Asset {asset_fname} has no checksum. Please set the checksum in the JSON file.") + + for host in hosts: + update_json_for_host(tmp_json_path, version, host, asset_url, asset_fname, asset_checksum, asset_size) + +def get_release_info(version): + url = f"https://api.github.com/repos/espressif/esptool/releases/tags/v{version}" + response = requests.get(url) + response.raise_for_status() + return response.json() + +def main(): + parser = argparse.ArgumentParser(description="Repack esptool and update JSON metadata.") + parser.add_argument("version", help="Version of the esptool (e.g. 5.0.dev1)") + parser.add_argument("-l", "--local", dest="base_folder", help="Enable local build mode and set the base folder with unpacked artifacts") + args = parser.parse_args() + + script_dir = Path(__file__).resolve().parent + json_path = (script_dir / "../../package/package_esp32_index.template.json").resolve() + tmp_json_path = Path(str(json_path) + ".tmp") + shutil.copy(json_path, tmp_json_path) + + local_build = args.base_folder is not None + + if local_build: + os.chdir(args.base_folder) + os.environ['COPYFILE_DISABLE'] = 'true' # this disables including resource forks in tar files on macOS + # Clear any existing archive files + for file in Path(args.base_folder).glob("esptool-*.*"): + file.unlink() + archive_files = create_archives(args.version, args.base_folder) + update_json_from_local_build(tmp_json_path, args.version, args.base_folder, archive_files) + else: + release_info = get_release_info(args.version) + update_json_from_release(tmp_json_path, args.version, release_info) + + print(f"Updating esptool version fields to {args.version}") + update_tools_dependencies(tmp_json_path, args.version) + + shutil.move(tmp_json_path, json_path) + print(f"Done. JSON updated at {json_path}") + +if __name__ == "__main__": + main() diff --git a/package/package_esp32_index.template.json b/package/package_esp32_index.template.json index bbb77f8ef5a..5f5d1f0e08b 100644 --- a/package/package_esp32_index.template.json +++ b/package/package_esp32_index.template.json @@ -81,7 +81,7 @@ { "packager": "esp32", "name": "esptool_py", - "version": "5.0.dev1" + "version": "5.0.0" }, { "packager": "esp32", @@ -469,56 +469,56 @@ }, { "name": "esptool_py", - "version": "5.0.dev1", + "version": "5.0.0", "systems": [ { "host": "aarch64-linux-gnu", - "url": "https://github.com/espressif/arduino-esp32/releases/download/3.2.0/esptool-v5.0.dev1-linux-aarch64.tar.gz", - "archiveFileName": "esptool-v5.0.dev1-linux-aarch64.tar.gz", - "checksum": "SHA-256:bfafa7a7723ebbabfd8b6e3ca5ae00bfead0331de923754aeddb43b2c116a078", - "size": "58241736" + "url": "https://github.com/espressif/esptool/releases/download/v5.0.0/esptool-v5.0.0-linux-aarch64.tar.gz", + "archiveFileName": "esptool-v5.0.0-linux-aarch64.tar.gz", + "checksum": "SHA-256:2bf239f3ed76141a957cadb205b94414ec6da9ace4e85f285e247d20a92b83e3", + "size": "58231895" }, { "host": "x86_64-pc-linux-gnu", - "url": "https://github.com/espressif/arduino-esp32/releases/download/3.2.0/esptool-v5.0.dev1-linux-amd64.tar.gz", - "archiveFileName": "esptool-v5.0.dev1-linux-amd64.tar.gz", - "checksum": "SHA-256:acd0486e96586b99d053a1479acbbbfcae8667227c831cdc53a171f9ccfa27ee", - "size": "100740042" + "url": "https://github.com/espressif/esptool/releases/download/v5.0.0/esptool-v5.0.0-linux-amd64.tar.gz", + "archiveFileName": "esptool-v5.0.0-linux-amd64.tar.gz", + "checksum": "SHA-256:3b3835d266ac61f3242758f2fe34e3b33dbe6ee4b5acde005da793356f9f7043", + "size": "100783748" }, { "host": "arm-linux-gnueabihf", - "url": "https://github.com/espressif/arduino-esp32/releases/download/3.2.0/esptool-v5.0.dev1-linux-armv7.tar.gz", - "archiveFileName": "esptool-v5.0.dev1-linux-armv7.tar.gz", - "checksum": "SHA-256:ea77a38681506761bbb7b0b39c130811ed565667b67ebbdb4d6dcc6cb6e07368", - "size": "53451939" + "url": "https://github.com/espressif/esptool/releases/download/v5.0.0/esptool-v5.0.0-linux-armv7.tar.gz", + "archiveFileName": "esptool-v5.0.0-linux-armv7.tar.gz", + "checksum": "SHA-256:e55cd321abecfcf27f72a2bff5d5e19a5365fd400de66d71c5e7218e77556315", + "size": "53461760" }, { "host": "x86_64-apple-darwin", - "url": "https://github.com/espressif/arduino-esp32/releases/download/3.2.0/esptool-v5.0.dev1-macos-amd64.tar.gz", - "archiveFileName": "esptool-v5.0.dev1-macos-amd64.tar.gz", - "checksum": "SHA-256:900a8e90731208bee96647e0e207a43612b9452c2120c4fdc0ff4c6be226257b", - "size": "59631998" + "url": "https://github.com/espressif/esptool/releases/download/v5.0.0/esptool-v5.0.0-macos-amd64.tar.gz", + "archiveFileName": "esptool-v5.0.0-macos-amd64.tar.gz", + "checksum": "SHA-256:424da2bdf0435257ad81bcb7eae6fd8dd7f675ce5b2ee60032f4ecec4d6a5d45", + "size": "59629533" }, { "host": "arm64-apple-darwin", - "url": "https://github.com/espressif/arduino-esp32/releases/download/3.2.0/esptool-v5.0.dev1-macos-arm64.tar.gz", - "archiveFileName": "esptool-v5.0.dev1-macos-arm64.tar.gz", - "checksum": "SHA-256:3653f4de73cb4fc6a25351eaf663708e91c65ae3265d75bd54ca4315a4350bb4", - "size": "56349992" + "url": "https://github.com/espressif/esptool/releases/download/v5.0.0/esptool-v5.0.0-macos-arm64.tar.gz", + "archiveFileName": "esptool-v5.0.0-macos-arm64.tar.gz", + "checksum": "SHA-256:b91dfe1da7b0041376683dec10a91dfb266fbda2fb86ed87c4a034ff7182ee56", + "size": "56343104" }, { "host": "x86_64-mingw32", - "url": "https://github.com/espressif/arduino-esp32/releases/download/3.2.0/esptool-v5.0.dev1-win64.zip", - "archiveFileName": "esptool-v5.0.dev1-win64.zip", - "checksum": "SHA-256:1e8fd89645daf94f2d4406ec73c9004e617ea921079515f9fd749205eece4d6d", - "size": "59102658" + "url": "https://github.com/espressif/esptool/releases/download/v5.0.0/esptool-v5.0.0-windows-amd64.zip", + "archiveFileName": "esptool-v5.0.0-windows-amd64.zip", + "checksum": "SHA-256:2294107f66db6f09b886b337728a981173c9e7eab45a030928a8a5a1370611ca", + "size": "59105322" }, { "host": "i686-mingw32", - "url": "https://github.com/espressif/arduino-esp32/releases/download/3.2.0/esptool-v5.0.dev1-win64.zip", - "archiveFileName": "esptool-v5.0.dev1-win64.zip", - "checksum": "SHA-256:1e8fd89645daf94f2d4406ec73c9004e617ea921079515f9fd749205eece4d6d", - "size": "59102658" + "url": "https://github.com/espressif/esptool/releases/download/v5.0.0/esptool-v5.0.0-windows-amd64.zip", + "archiveFileName": "esptool-v5.0.0-windows-amd64.zip", + "checksum": "SHA-256:2294107f66db6f09b886b337728a981173c9e7eab45a030928a8a5a1370611ca", + "size": "59105322" } ] }, From ccc0a69ef88202519c0773ec254ee31cbe2baf82 Mon Sep 17 00:00:00 2001 From: Lucas Saavedra Vaz <32426024+lucasssvaz@users.noreply.github.com> Date: Tue, 8 Jul 2025 11:58:36 -0300 Subject: [PATCH 103/173] change(esptool): Upgrade esptool to release v5.0.0 (#11563) --- .github/scripts/package_esptool.sh | 129 ---------------------- .github/scripts/update_esptool.py | 2 +- package/package_esp32_index.template.json | 60 +++++----- 3 files changed, 31 insertions(+), 160 deletions(-) delete mode 100755 .github/scripts/package_esptool.sh diff --git a/.github/scripts/package_esptool.sh b/.github/scripts/package_esptool.sh deleted file mode 100755 index 32b87b277e9..00000000000 --- a/.github/scripts/package_esptool.sh +++ /dev/null @@ -1,129 +0,0 @@ -#!/bin/bash - -set -euo pipefail - -# Check version argument -if [[ $# -ne 3 ]]; then - echo "Usage: $0 " - echo "Example: $0 5.0.dev1 /tmp/esptool /tmp/esptool-5.0.dev1.json" - exit 1 -fi - -VERSION=$1 -BASE_FOLDER=$2 -JSON_PATH=$3 - -export COPYFILE_DISABLE=1 - -shopt -s nullglob # So for loop doesn't run if no matches - -# Function to update JSON for a given host -function update_json_for_host { - local host=$1 - local archive=$2 - - # Extract the old url from the JSON for this host, then replace only the filename - old_url=$(jq -r --arg host "$host" ' - .packages[].tools[] | select(.name == "esptool_py") | .systems[] | select(.host == $host) | .url // empty - ' "$tmp_json") - if [[ -n "$old_url" ]]; then - base_url="${old_url%/*}" - url="$base_url/$archive" - else - echo "No old url found for $host" - exit 1 - fi - - archiveFileName="$archive" - checksum="SHA-256:$(shasum -a 256 "$archive" | awk '{print $1}')" - size=$(stat -f%z "$archive") - - # Use jq to update the JSON - jq --arg host "$host" \ - --arg url "$url" \ - --arg archiveFileName "$archiveFileName" \ - --arg checksum "$checksum" \ - --arg size "$size" \ - ' - .packages[].tools[] - |= if .name == "esptool_py" then - .systems = ( - ((.systems // []) | map(select(.host != $host))) + [{ - host: $host, - url: $url, - archiveFileName: $archiveFileName, - checksum: $checksum, - size: $size - }] - ) - else - . - end - ' "$tmp_json" > "$tmp_json.new" && mv "$tmp_json.new" "$tmp_json" -} - -cd "$BASE_FOLDER" - -# Delete all archives before starting -rm -f esptool-*.tar.gz esptool-*.zip - -for dir in esptool-*; do - # Check if directory exists and is a directory - if [[ ! -d "$dir" ]]; then - continue - fi - - base="${dir#esptool-}" - - # Add 'linux-' prefix if base doesn't contain linux/macos/win64 - if [[ "$base" != *linux* && "$base" != *macos* && "$base" != *win64* ]]; then - base="linux-${base}" - fi - - if [[ "$dir" == esptool-win* ]]; then - # Windows zip archive - zipfile="esptool-v${VERSION}-${base}.zip" - echo "Creating $zipfile from $dir ..." - zip -r "$zipfile" "$dir" - else - # Non-Windows: set permissions and tar.gz archive - tarfile="esptool-v${VERSION}-${base}.tar.gz" - echo "Setting permissions and creating $tarfile from $dir ..." - chmod -R u=rwx,g=rx,o=rx "$dir" - tar -cvzf "$tarfile" "$dir" - fi -done - -# After the for loop, update the JSON for each archive -# Create a temporary JSON file to accumulate changes -tmp_json="${JSON_PATH}.tmp" -cp "$JSON_PATH" "$tmp_json" - -for archive in esptool-v"${VERSION}"-*.tar.gz esptool-v"${VERSION}"-*.zip; do - [ -f "$archive" ] || continue - - echo "Updating JSON for $archive" - - # Determine host from archive name - case "$archive" in - *linux-amd64*) host="x86_64-pc-linux-gnu" ;; - *linux-armv7*) host="arm-linux-gnueabihf" ;; - *linux-aarch64*) host="aarch64-linux-gnu" ;; - *macos-amd64*) host="x86_64-apple-darwin" ;; - *macos-arm64*) host="arm64-apple-darwin" ;; - *win64*) hosts=("x86_64-mingw32" "i686-mingw32") ;; - *) echo "Unknown host for $archive"; continue ;; - esac - - # For win64, loop over both hosts; otherwise, use a single host - if [[ "$archive" == *win64* ]]; then - for host in "${hosts[@]}"; do - update_json_for_host "$host" "$archive" - done - else - update_json_for_host "$host" "$archive" - fi -done - -# After all archives are processed, move the temporary JSON to the final file -mv "$tmp_json" "$JSON_PATH" diff --git a/.github/scripts/update_esptool.py b/.github/scripts/update_esptool.py index dd5de5526c3..d99462fcb8f 100644 --- a/.github/scripts/update_esptool.py +++ b/.github/scripts/update_esptool.py @@ -186,7 +186,7 @@ def update_json_from_release(tmp_json_path, version, release_info): continue asset_url = asset.get("browser_download_url") - asset_checksum = asset.get("digest") + asset_checksum = asset.get("digest").replace("sha256:", "SHA-256:") asset_size = asset.get("size") if asset_checksum is None: asset_checksum = "" diff --git a/package/package_esp32_index.template.json b/package/package_esp32_index.template.json index a02be509094..9c2c754504e 100644 --- a/package/package_esp32_index.template.json +++ b/package/package_esp32_index.template.json @@ -81,7 +81,7 @@ { "packager": "esp32", "name": "esptool_py", - "version": "5.0.dev1" + "version": "5.0.0" }, { "packager": "esp32", @@ -469,56 +469,56 @@ }, { "name": "esptool_py", - "version": "5.0.dev1", + "version": "5.0.0", "systems": [ { "host": "aarch64-linux-gnu", - "url": "https://github.com/espressif/arduino-esp32/releases/download/3.2.0/esptool-v5.0.dev1-linux-aarch64.tar.gz", - "archiveFileName": "esptool-v5.0.dev1-linux-aarch64.tar.gz", - "checksum": "SHA-256:bfafa7a7723ebbabfd8b6e3ca5ae00bfead0331de923754aeddb43b2c116a078", - "size": "58241736" + "url": "https://github.com/espressif/esptool/releases/download/v5.0.0/esptool-v5.0.0-linux-aarch64.tar.gz", + "archiveFileName": "esptool-v5.0.0-linux-aarch64.tar.gz", + "checksum": "SHA-256:2bf239f3ed76141a957cadb205b94414ec6da9ace4e85f285e247d20a92b83e3", + "size": "58231895" }, { "host": "x86_64-pc-linux-gnu", - "url": "https://github.com/espressif/arduino-esp32/releases/download/3.2.0/esptool-v5.0.dev1-linux-amd64.tar.gz", - "archiveFileName": "esptool-v5.0.dev1-linux-amd64.tar.gz", - "checksum": "SHA-256:acd0486e96586b99d053a1479acbbbfcae8667227c831cdc53a171f9ccfa27ee", - "size": "100740042" + "url": "https://github.com/espressif/esptool/releases/download/v5.0.0/esptool-v5.0.0-linux-amd64.tar.gz", + "archiveFileName": "esptool-v5.0.0-linux-amd64.tar.gz", + "checksum": "SHA-256:3b3835d266ac61f3242758f2fe34e3b33dbe6ee4b5acde005da793356f9f7043", + "size": "100783748" }, { "host": "arm-linux-gnueabihf", - "url": "https://github.com/espressif/arduino-esp32/releases/download/3.2.0/esptool-v5.0.dev1-linux-armv7.tar.gz", - "archiveFileName": "esptool-v5.0.dev1-linux-armv7.tar.gz", - "checksum": "SHA-256:ea77a38681506761bbb7b0b39c130811ed565667b67ebbdb4d6dcc6cb6e07368", - "size": "53451939" + "url": "https://github.com/espressif/esptool/releases/download/v5.0.0/esptool-v5.0.0-linux-armv7.tar.gz", + "archiveFileName": "esptool-v5.0.0-linux-armv7.tar.gz", + "checksum": "SHA-256:e55cd321abecfcf27f72a2bff5d5e19a5365fd400de66d71c5e7218e77556315", + "size": "53461760" }, { "host": "x86_64-apple-darwin", - "url": "https://github.com/espressif/arduino-esp32/releases/download/3.2.0/esptool-v5.0.dev1-macos-amd64.tar.gz", - "archiveFileName": "esptool-v5.0.dev1-macos-amd64.tar.gz", - "checksum": "SHA-256:900a8e90731208bee96647e0e207a43612b9452c2120c4fdc0ff4c6be226257b", - "size": "59631998" + "url": "https://github.com/espressif/esptool/releases/download/v5.0.0/esptool-v5.0.0-macos-amd64.tar.gz", + "archiveFileName": "esptool-v5.0.0-macos-amd64.tar.gz", + "checksum": "SHA-256:424da2bdf0435257ad81bcb7eae6fd8dd7f675ce5b2ee60032f4ecec4d6a5d45", + "size": "59629533" }, { "host": "arm64-apple-darwin", - "url": "https://github.com/espressif/arduino-esp32/releases/download/3.2.0/esptool-v5.0.dev1-macos-arm64.tar.gz", - "archiveFileName": "esptool-v5.0.dev1-macos-arm64.tar.gz", - "checksum": "SHA-256:3653f4de73cb4fc6a25351eaf663708e91c65ae3265d75bd54ca4315a4350bb4", - "size": "56349992" + "url": "https://github.com/espressif/esptool/releases/download/v5.0.0/esptool-v5.0.0-macos-arm64.tar.gz", + "archiveFileName": "esptool-v5.0.0-macos-arm64.tar.gz", + "checksum": "SHA-256:b91dfe1da7b0041376683dec10a91dfb266fbda2fb86ed87c4a034ff7182ee56", + "size": "56343104" }, { "host": "x86_64-mingw32", - "url": "https://github.com/espressif/arduino-esp32/releases/download/3.2.0/esptool-v5.0.dev1-win64.zip", - "archiveFileName": "esptool-v5.0.dev1-win64.zip", - "checksum": "SHA-256:1e8fd89645daf94f2d4406ec73c9004e617ea921079515f9fd749205eece4d6d", - "size": "59102658" + "url": "https://github.com/espressif/esptool/releases/download/v5.0.0/esptool-v5.0.0-windows-amd64.zip", + "archiveFileName": "esptool-v5.0.0-windows-amd64.zip", + "checksum": "SHA-256:2294107f66db6f09b886b337728a981173c9e7eab45a030928a8a5a1370611ca", + "size": "59105322" }, { "host": "i686-mingw32", - "url": "https://github.com/espressif/arduino-esp32/releases/download/3.2.0/esptool-v5.0.dev1-win64.zip", - "archiveFileName": "esptool-v5.0.dev1-win64.zip", - "checksum": "SHA-256:1e8fd89645daf94f2d4406ec73c9004e617ea921079515f9fd749205eece4d6d", - "size": "59102658" + "url": "https://github.com/espressif/esptool/releases/download/v5.0.0/esptool-v5.0.0-windows-amd64.zip", + "archiveFileName": "esptool-v5.0.0-windows-amd64.zip", + "checksum": "SHA-256:2294107f66db6f09b886b337728a981173c9e7eab45a030928a8a5a1370611ca", + "size": "59105322" } ] }, From ee021855a156847dfcf75a9aeb8d585213f09e42 Mon Sep 17 00:00:00 2001 From: Lucas Saavedra Vaz <32426024+lucasssvaz@users.noreply.github.com> Date: Wed, 9 Jul 2025 07:24:42 -0300 Subject: [PATCH 104/173] feat(docs_version): Update docs in update-version script (#11564) * feat(docs_version): Update docs in update-version script * fix(logging): Fix log message * fix(idf_version): Add error if IDF version is not found --- .github/scripts/update-version.sh | 14 ++++++++++++++ docs/conf_common.py | 2 +- 2 files changed, 15 insertions(+), 1 deletion(-) diff --git a/.github/scripts/update-version.sh b/.github/scripts/update-version.sh index 9a38b27a57a..622f2fe8ff8 100755 --- a/.github/scripts/update-version.sh +++ b/.github/scripts/update-version.sh @@ -1,4 +1,5 @@ #!/bin/bash +# Disable shellcheck warning about using 'cat' to read a file. # shellcheck disable=SC2002 # For reference: add tools for all boards by replacing one line in each board @@ -23,7 +24,15 @@ ESP_ARDUINO_VERSION_MINOR="$2" ESP_ARDUINO_VERSION_PATCH="$3" ESP_ARDUINO_VERSION="$ESP_ARDUINO_VERSION_MAJOR.$ESP_ARDUINO_VERSION_MINOR.$ESP_ARDUINO_VERSION_PATCH" +# Get ESP-IDF version from push.yml (this way we can ensure that the version is correct even if the local libs are not up to date) +ESP_IDF_VERSION=$(grep "idf_ver:" .github/workflows/push.yml | sed 's/.*release-v\([^"]*\).*/\1/') +if [ -z "$ESP_IDF_VERSION" ]; then + echo "Error: ESP-IDF version not found in push.yml" >&2 + exit 1 +fi + echo "New Arduino Version: $ESP_ARDUINO_VERSION" +echo "ESP-IDF Version: $ESP_IDF_VERSION" echo "Updating platform.txt..." cat platform.txt | sed "s/version=.*/version=$ESP_ARDUINO_VERSION/g" > __platform.txt && mv __platform.txt platform.txt @@ -31,6 +40,11 @@ cat platform.txt | sed "s/version=.*/version=$ESP_ARDUINO_VERSION/g" > __platfor echo "Updating package.json..." cat package.json | sed "s/.*\"version\":.*/ \"version\": \"$ESP_ARDUINO_VERSION\",/g" > __package.json && mv __package.json package.json +echo "Updating docs/conf_common.py..." +cat docs/conf_common.py | \ +sed "s/.. |version| replace:: .*/.. |version| replace:: $ESP_ARDUINO_VERSION/g" | \ +sed "s/.. |idf_version| replace:: .*/.. |idf_version| replace:: $ESP_IDF_VERSION/g" > docs/__conf_common.py && mv docs/__conf_common.py docs/conf_common.py + echo "Updating cores/esp32/esp_arduino_version.h..." cat cores/esp32/esp_arduino_version.h | \ sed "s/#define ESP_ARDUINO_VERSION_MAJOR.*/#define ESP_ARDUINO_VERSION_MAJOR $ESP_ARDUINO_VERSION_MAJOR/g" | \ diff --git a/docs/conf_common.py b/docs/conf_common.py index 6945c0d190d..af1d615f753 100644 --- a/docs/conf_common.py +++ b/docs/conf_common.py @@ -4,7 +4,7 @@ # Used for substituting variables in the documentation rst_prolog = """ -.. |version| replace:: 3.2.0 +.. |version| replace:: 3.2.1 .. |idf_version| replace:: 5.4 """ From 87b718a59cbeed010eee2c6a79004e22f71564ac Mon Sep 17 00:00:00 2001 From: Lucas Saavedra Vaz <32426024+lucasssvaz@users.noreply.github.com> Date: Wed, 9 Jul 2025 07:25:16 -0300 Subject: [PATCH 105/173] fix(merge): Fix merging CN Json (#11574) --- .github/scripts/merge_packages.py | 16 ++++++++-------- .github/scripts/release_append_cn.py | 5 +++-- 2 files changed, 11 insertions(+), 10 deletions(-) diff --git a/.github/scripts/merge_packages.py b/.github/scripts/merge_packages.py index 7e4f47ca8b3..8d1f200ec5c 100755 --- a/.github/scripts/merge_packages.py +++ b/.github/scripts/merge_packages.py @@ -4,6 +4,7 @@ # Usage: # python merge_packages.py package_esp8266com_index.json version/new/package_esp8266com_index.json # Written by Ivan Grokhotkov, 2015 +# Updated by lucasssvaz to handle Chinese version sorting, 2025 # from __future__ import print_function @@ -36,20 +37,19 @@ def merge_objects(versions, obj): # Normalize ESP release version string (x.x.x) by adding '-rc' (x.x.x-rc9223372036854775807) -# to ensure having REL above any RC +# to ensure having REL above any RC. CN version will be sorted after the official version if they happen +# to be mixed (normally, CN and non-CN versions should not be mixed) # Dummy approach, functional anyway for current ESP package versioning # (unlike NormalizedVersion/LooseVersion/StrictVersion & similar crap) def pkgVersionNormalized(versionString): - - verStr = str(versionString) + verStr = str(versionString).replace("-cn", "") verParts = re.split(r"\.|-rc|-alpha", verStr, flags=re.IGNORECASE) if len(verParts) == 3: - if sys.version_info > (3, 0): # Python 3 - verStr = str(versionString) + "-rc" + str(sys.maxsize) - else: # Python 2 - verStr = str(versionString) + "-rc" + str(sys.maxint) - + if "-cn" in str(versionString): + verStr = verStr + "-rc" + str(sys.maxsize // 2) + else: + verStr = verStr + "-rc" + str(sys.maxsize) elif len(verParts) != 4: print("pkgVersionNormalized WARNING: unexpected version format: {0})".format(verStr), file=sys.stderr) diff --git a/.github/scripts/release_append_cn.py b/.github/scripts/release_append_cn.py index b29fe0c31ba..2342834bb7e 100755 --- a/.github/scripts/release_append_cn.py +++ b/.github/scripts/release_append_cn.py @@ -17,8 +17,9 @@ def append_cn_to_versions(obj): if isinstance(obj, dict): - # dfu-util comes from arduino.cc and not from the Chinese mirrors, so we skip it - if obj.get("name") == "dfu-util": + # Skip tools that are not from the esp32 package + packager = obj.get("packager") + if packager is not None and packager != "esp32": return for key, value in obj.items(): From 4ee17dea045b2b4f8bd5965bdde6e9110b612026 Mon Sep 17 00:00:00 2001 From: Sugar Glider Date: Fri, 11 Jul 2025 11:25:27 -0300 Subject: [PATCH 106/173] feat(matter): new matter lambda function example (#11561) Adds a new Matter Library example using lambda function to creat 6 endpoints using a single callback --- .../MatterLambdaSingleCallbackManyEPs.ino | 112 ++++++++++++++++++ .../MatterLambdaSingleCallbackManyEPs/ci.json | 7 ++ libraries/Matter/keywords.txt | 6 + 3 files changed, 125 insertions(+) create mode 100644 libraries/Matter/examples/MatterLambdaSingleCallbackManyEPs/MatterLambdaSingleCallbackManyEPs.ino create mode 100644 libraries/Matter/examples/MatterLambdaSingleCallbackManyEPs/ci.json diff --git a/libraries/Matter/examples/MatterLambdaSingleCallbackManyEPs/MatterLambdaSingleCallbackManyEPs.ino b/libraries/Matter/examples/MatterLambdaSingleCallbackManyEPs/MatterLambdaSingleCallbackManyEPs.ino new file mode 100644 index 00000000000..4992771d925 --- /dev/null +++ b/libraries/Matter/examples/MatterLambdaSingleCallbackManyEPs/MatterLambdaSingleCallbackManyEPs.ino @@ -0,0 +1,112 @@ +// Copyright 2025 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. + +/* + This example creates 6 on-off light endpoints that share the same onChangeOnOff() callback code. + It uses Lambda Function with an extra Lambda Capture information that links the Endpoint to its individual information. + After the Matter example is commissioned, the expected Serial output shall be similar to this: + +Matter App Control: 'Room 1' (OnOffLight[0], Endpoint 1, GPIO 2) changed to: OFF +Matter App Control: 'Room 1' (OnOffLight[0], Endpoint 1, GPIO 2) changed to: ON +Matter App Control: 'Room 5' (OnOffLight[4], Endpoint 5, GPIO 10) changed to: ON +Matter App Control: 'Room 2' (OnOffLight[1], Endpoint 2, GPIO 4) changed to: ON +Matter App Control: 'Room 4' (OnOffLight[3], Endpoint 4, GPIO 8) changed to: ON +Matter App Control: 'Room 6' (OnOffLight[5], Endpoint 6, GPIO 12) changed to: ON +Matter App Control: 'Room 3' (OnOffLight[2], Endpoint 3, GPIO 6) changed to: ON +Matter App Control: 'Room 5' (OnOffLight[4], Endpoint 5, GPIO 10) changed to: OFF +*/ + +// Matter Manager +#include +#include + +// WiFi is manually set and started +const char *ssid = "your-ssid"; // Change this to your WiFi SSID +const char *password = "your-password"; // Change this to your WiFi password + +//number of On-Off Lights: +const uint8_t MAX_LIGHT_NUMBER = 6; + +// array of OnOffLight endpoints +MatterOnOffLight OnOffLight[MAX_LIGHT_NUMBER]; + +// all pins, one for each on-off light +uint8_t lightPins[MAX_LIGHT_NUMBER] = {2, 4, 6, 8, 10, 12}; // must replace it by the real pin for the target SoC and application + +// friendly OnOffLights names used for printing a message in the callback +const char *lightName[MAX_LIGHT_NUMBER] = { + "Room 1", "Room 2", "Room 3", "Room 4", "Room 5", "Room 6", +}; + +// simple setup() function +void setup() { + Serial.begin(115200); // callback will just print a message in the console + + // We start by connecting to a WiFi network + Serial.print("Connecting to "); + Serial.println(ssid); + // Manually connect to WiFi + WiFi.begin(ssid, password); + // Wait for connection + while (WiFi.status() != WL_CONNECTED) { + delay(500); + Serial.print("."); + } + Serial.println("\r\nWiFi connected"); + Serial.println("IP address: "); + Serial.println(WiFi.localIP()); + delay(500); + + // setup all the OnOff Light endpoint and their lambda callback functions + for (uint8_t i = 0; i < MAX_LIGHT_NUMBER; i++) { + pinMode(lightPins[i], OUTPUT); // set the GPIO function + OnOffLight[i].begin(false); // off + + // inline lambda function using capture array index -> it will just print a message in the console + OnOffLight[i].onChangeOnOff([i](bool state) -> bool { + // Display message with the specific light name and details + Serial.printf( + "Matter App Control: '%s' (OnOffLight[%d], Endpoint %d, GPIO %d) changed to: %s\r\n", lightName[i], i, OnOffLight[i].getEndPointId(), lightPins[i], + state ? "ON" : "OFF" + ); + + return true; + }); + } + // last step, starting Matter Stack + Matter.begin(); +} + +void loop() { + // Check Matter Plugin Commissioning state, which may change during execution of loop() + if (!Matter.isDeviceCommissioned()) { + Serial.println(""); + Serial.println("Matter Node is not commissioned yet."); + Serial.println("Initiate the device discovery in your Matter environment."); + Serial.println("Commission it to your Matter hub with the manual pairing code or QR code"); + Serial.printf("Manual pairing code: %s\r\n", Matter.getManualPairingCode().c_str()); + Serial.printf("QR code URL: %s\r\n", Matter.getOnboardingQRCodeUrl().c_str()); + // waits for Matter Plugin Commissioning. + uint32_t timeCount = 0; + while (!Matter.isDeviceCommissioned()) { + delay(100); + if ((timeCount++ % 50) == 0) { // 50*100ms = 5 sec + Serial.println("Matter Node not commissioned yet. Waiting for commissioning."); + } + } + Serial.println("Matter Node is commissioned and connected to the WiFi network. Ready for use."); + } + + delay(500); +} diff --git a/libraries/Matter/examples/MatterLambdaSingleCallbackManyEPs/ci.json b/libraries/Matter/examples/MatterLambdaSingleCallbackManyEPs/ci.json new file mode 100644 index 00000000000..556a8a9ee6b --- /dev/null +++ b/libraries/Matter/examples/MatterLambdaSingleCallbackManyEPs/ci.json @@ -0,0 +1,7 @@ +{ + "fqbn_append": "PartitionScheme=huge_app", + "requires": [ + "CONFIG_SOC_WIFI_SUPPORTED=y", + "CONFIG_ESP_MATTER_ENABLE_DATA_MODEL=y" + ] +} diff --git a/libraries/Matter/keywords.txt b/libraries/Matter/keywords.txt index 68aaebb1d4d..edba06083bd 100644 --- a/libraries/Matter/keywords.txt +++ b/libraries/Matter/keywords.txt @@ -36,8 +36,10 @@ EndPointSpeedCB KEYWORD1 EndPointOnOffCB KEYWORD1 EndPointBrightnessCB KEYWORD1 EndPointRGBColorCB KEYWORD1 +EndPointIdentifyCB KEYWORD1 matterEvent_t KEYWORD1 matterEventCB KEYWORD1 +attrOperation_t KEYWORD1 ####################################### # Methods and Functions (KEYWORD2) @@ -111,6 +113,10 @@ onChangeLocalTemperature KEYWORD2 onChangeCoolingSetpoint KEYWORD2 onChangeHeatingSetpoint KEYWORD2 onEvent KEYWORD2 +setEndPointId KEYWORD2 +getEndPointId KEYWORD2 +onIdentify KEYWORD2 +endpointIdentifyCB KEYWORD2 ####################################### # Constants (LITERAL1) From c6a3bcb014c7fff53e6c1c41fec50e3ec4de8514 Mon Sep 17 00:00:00 2001 From: Sugar Glider Date: Sat, 12 Jul 2025 00:11:41 -0300 Subject: [PATCH 107/173] feat(matter): removing wifi requirement for H2 and C5 (#11581) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit This PR removes WiFi provisioning support from CI and examples (shifting to Thread/BLE provisioning), updates the CI configs for all Matter examples to drop the WiFi requirement, and adds new API keywords and a fresh “LambdaSingleCallbackManyEPs” example. - Deleted the WiFiProvWithinMatter example (its .ino and ci.json) since BLE is now used for provisioning. - Stripped "CONFIG_SOC_WIFI_SUPPORTED=y" from the CI JSON of existing examples to test Thread-only builds. - Updated keywords.txt with new Matter API identifiers and introduced a new “LambdaSingleCallbackManyEPs” example with CI and source --- .../Matter/examples/MatterColorLight/ci.json | 11 +- .../examples/MatterCommissionTest/ci.json | 1 - .../examples/MatterComposedLights/ci.json | 1 - .../examples/MatterContactSensor/ci.json | 1 - .../examples/MatterDimmableLight/ci.json | 1 - .../examples/MatterEnhancedColorLight/ci.json | 9 +- .../Matter/examples/MatterEvents/ci.json | 1 - libraries/Matter/examples/MatterFan/ci.json | 9 +- .../examples/MatterHumiditySensor/ci.json | 1 - .../MatterLambdaSingleCallbackManyEPs.ino | 126 +++++++++++++++ .../MatterLambdaSingleCallbackManyEPs/ci.json | 6 + .../Matter/examples/MatterMinimum/ci.json | 1 - .../examples/MatterOccupancySensor/ci.json | 1 - .../Matter/examples/MatterOnIdentify/ci.json | 1 - .../Matter/examples/MatterOnOffLight/ci.json | 1 - .../Matter/examples/MatterOnOffPlugin/ci.json | 1 - .../examples/MatterPressureSensor/ci.json | 1 - .../Matter/examples/MatterSmartButon/ci.json | 1 - .../examples/MatterTemperatureLight/ci.json | 1 - .../examples/MatterTemperatureSensor/ci.json | 1 - .../Matter/examples/MatterThermostat/ci.json | 1 - .../WiFiProvWithinMatter.ino | 152 ------------------ .../examples/WiFiProvWithinMatter/ci.json | 7 - libraries/Matter/keywords.txt | 8 + 24 files changed, 153 insertions(+), 191 deletions(-) create mode 100644 libraries/Matter/examples/MatterLambdaSingleCallbackManyEPs/MatterLambdaSingleCallbackManyEPs.ino create mode 100644 libraries/Matter/examples/MatterLambdaSingleCallbackManyEPs/ci.json delete mode 100644 libraries/Matter/examples/WiFiProvWithinMatter/WiFiProvWithinMatter.ino delete mode 100644 libraries/Matter/examples/WiFiProvWithinMatter/ci.json diff --git a/libraries/Matter/examples/MatterColorLight/ci.json b/libraries/Matter/examples/MatterColorLight/ci.json index d5f63487506..90b393f9156 100644 --- a/libraries/Matter/examples/MatterColorLight/ci.json +++ b/libraries/Matter/examples/MatterColorLight/ci.json @@ -1,7 +1,6 @@ { - "fqbn_append": "PartitionScheme=huge_app", - "requires": [ - "CONFIG_SOC_WIFI_SUPPORTED=y", - "CONFIG_ESP_MATTER_ENABLE_DATA_MODEL=y" - ] - } + "fqbn_append": "PartitionScheme=huge_app", + "requires": [ + "CONFIG_ESP_MATTER_ENABLE_DATA_MODEL=y" + ] +} diff --git a/libraries/Matter/examples/MatterCommissionTest/ci.json b/libraries/Matter/examples/MatterCommissionTest/ci.json index 556a8a9ee6b..90b393f9156 100644 --- a/libraries/Matter/examples/MatterCommissionTest/ci.json +++ b/libraries/Matter/examples/MatterCommissionTest/ci.json @@ -1,7 +1,6 @@ { "fqbn_append": "PartitionScheme=huge_app", "requires": [ - "CONFIG_SOC_WIFI_SUPPORTED=y", "CONFIG_ESP_MATTER_ENABLE_DATA_MODEL=y" ] } diff --git a/libraries/Matter/examples/MatterComposedLights/ci.json b/libraries/Matter/examples/MatterComposedLights/ci.json index 556a8a9ee6b..90b393f9156 100644 --- a/libraries/Matter/examples/MatterComposedLights/ci.json +++ b/libraries/Matter/examples/MatterComposedLights/ci.json @@ -1,7 +1,6 @@ { "fqbn_append": "PartitionScheme=huge_app", "requires": [ - "CONFIG_SOC_WIFI_SUPPORTED=y", "CONFIG_ESP_MATTER_ENABLE_DATA_MODEL=y" ] } diff --git a/libraries/Matter/examples/MatterContactSensor/ci.json b/libraries/Matter/examples/MatterContactSensor/ci.json index 556a8a9ee6b..90b393f9156 100644 --- a/libraries/Matter/examples/MatterContactSensor/ci.json +++ b/libraries/Matter/examples/MatterContactSensor/ci.json @@ -1,7 +1,6 @@ { "fqbn_append": "PartitionScheme=huge_app", "requires": [ - "CONFIG_SOC_WIFI_SUPPORTED=y", "CONFIG_ESP_MATTER_ENABLE_DATA_MODEL=y" ] } diff --git a/libraries/Matter/examples/MatterDimmableLight/ci.json b/libraries/Matter/examples/MatterDimmableLight/ci.json index 556a8a9ee6b..90b393f9156 100644 --- a/libraries/Matter/examples/MatterDimmableLight/ci.json +++ b/libraries/Matter/examples/MatterDimmableLight/ci.json @@ -1,7 +1,6 @@ { "fqbn_append": "PartitionScheme=huge_app", "requires": [ - "CONFIG_SOC_WIFI_SUPPORTED=y", "CONFIG_ESP_MATTER_ENABLE_DATA_MODEL=y" ] } diff --git a/libraries/Matter/examples/MatterEnhancedColorLight/ci.json b/libraries/Matter/examples/MatterEnhancedColorLight/ci.json index 0665800b12b..90b393f9156 100644 --- a/libraries/Matter/examples/MatterEnhancedColorLight/ci.json +++ b/libraries/Matter/examples/MatterEnhancedColorLight/ci.json @@ -1,7 +1,6 @@ { - "fqbn_append": "PartitionScheme=huge_app", - "requires": [ - "CONFIG_SOC_WIFI_SUPPORTED=y", - "CONFIG_ESP_MATTER_ENABLE_DATA_MODEL=y" - ] + "fqbn_append": "PartitionScheme=huge_app", + "requires": [ + "CONFIG_ESP_MATTER_ENABLE_DATA_MODEL=y" + ] } diff --git a/libraries/Matter/examples/MatterEvents/ci.json b/libraries/Matter/examples/MatterEvents/ci.json index 556a8a9ee6b..90b393f9156 100644 --- a/libraries/Matter/examples/MatterEvents/ci.json +++ b/libraries/Matter/examples/MatterEvents/ci.json @@ -1,7 +1,6 @@ { "fqbn_append": "PartitionScheme=huge_app", "requires": [ - "CONFIG_SOC_WIFI_SUPPORTED=y", "CONFIG_ESP_MATTER_ENABLE_DATA_MODEL=y" ] } diff --git a/libraries/Matter/examples/MatterFan/ci.json b/libraries/Matter/examples/MatterFan/ci.json index 0665800b12b..90b393f9156 100644 --- a/libraries/Matter/examples/MatterFan/ci.json +++ b/libraries/Matter/examples/MatterFan/ci.json @@ -1,7 +1,6 @@ { - "fqbn_append": "PartitionScheme=huge_app", - "requires": [ - "CONFIG_SOC_WIFI_SUPPORTED=y", - "CONFIG_ESP_MATTER_ENABLE_DATA_MODEL=y" - ] + "fqbn_append": "PartitionScheme=huge_app", + "requires": [ + "CONFIG_ESP_MATTER_ENABLE_DATA_MODEL=y" + ] } diff --git a/libraries/Matter/examples/MatterHumiditySensor/ci.json b/libraries/Matter/examples/MatterHumiditySensor/ci.json index 556a8a9ee6b..90b393f9156 100644 --- a/libraries/Matter/examples/MatterHumiditySensor/ci.json +++ b/libraries/Matter/examples/MatterHumiditySensor/ci.json @@ -1,7 +1,6 @@ { "fqbn_append": "PartitionScheme=huge_app", "requires": [ - "CONFIG_SOC_WIFI_SUPPORTED=y", "CONFIG_ESP_MATTER_ENABLE_DATA_MODEL=y" ] } diff --git a/libraries/Matter/examples/MatterLambdaSingleCallbackManyEPs/MatterLambdaSingleCallbackManyEPs.ino b/libraries/Matter/examples/MatterLambdaSingleCallbackManyEPs/MatterLambdaSingleCallbackManyEPs.ino new file mode 100644 index 00000000000..c60cadd784f --- /dev/null +++ b/libraries/Matter/examples/MatterLambdaSingleCallbackManyEPs/MatterLambdaSingleCallbackManyEPs.ino @@ -0,0 +1,126 @@ +// Copyright 2025 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. + +/* + This example create 6 on-off light endpoint that share the same onChangeOnOff() callback code. + It uses Lambda Function with an extra Lambda Capture information that links the Endpoint to its individual information. + After the Matter example is commissioned, the expected Serial output shall be similar to this: + +Matter App Control: 'Room 1' (OnOffLight[0], Endpoint 1, GPIO 2) changed to: OFF +Matter App Control: 'Room 1' (OnOffLight[0], Endpoint 1, GPIO 2) changed to: ON +Matter App Control: 'Room 5' (OnOffLight[4], Endpoint 5, GPIO 10) changed to: ON +Matter App Control: 'Room 2' (OnOffLight[1], Endpoint 2, GPIO 4) changed to: ON +Matter App Control: 'Room 4' (OnOffLight[3], Endpoint 4, GPIO 8) changed to: ON +Matter App Control: 'Room 6' (OnOffLight[5], Endpoint 6, GPIO 12) changed to: ON +Matter App Control: 'Room 3' (OnOffLight[2], Endpoint 3, GPIO 6) changed to: ON +Matter App Control: 'Room 5' (OnOffLight[4], Endpoint 5, GPIO 10) changed to: OFF +*/ + +// Matter Manager +#include +#include +#if !CONFIG_ENABLE_CHIPOBLE +// if the device can be commissioned using BLE, WiFi is not used - save flash space +#include +// CONFIG_ENABLE_CHIPOBLE is enabled when BLE is used to commission the Matter Network +// WiFi is manually set and started +const char *ssid = "your-ssid"; // Change this to your WiFi SSID +const char *password = "your-password"; // Change this to your WiFi password +#endif + +//number of On-Off Lights: +const uint8_t MAX_LIGHT_NUMBER = 6; + +// array of OnOffLight endpoints +MatterOnOffLight OnOffLight[MAX_LIGHT_NUMBER]; + +// all pins, one for each on-off light +uint8_t lightPins[MAX_LIGHT_NUMBER] = {2, 4, 6, 8, 10, 12}; // must replace it by the real pin for the target SoC and application + +// friendly OnOffLights names used for printing a message in the callback +const char *lightName[MAX_LIGHT_NUMBER] = { + "Room 1", "Room 2", "Room 3", "Room 4", "Room 5", "Room 6", +}; + +// simple setup() function +void setup() { + Serial.begin(115200); // callback will just print a message in the console + +// CONFIG_ENABLE_CHIPOBLE is enabled when BLE is used to commission the Matter Network +#if !CONFIG_ENABLE_CHIPOBLE + // We start by connecting to a WiFi network + Serial.print("Connecting to "); + Serial.println(ssid); + // Manually connect to WiFi + WiFi.begin(ssid, password); + // Wait for connection + while (WiFi.status() != WL_CONNECTED) { + delay(500); + Serial.print("."); + } + Serial.println("\r\nWiFi connected"); + Serial.println("IP address: "); + Serial.println(WiFi.localIP()); + delay(500); +#endif + + // setup all the OnOff Light endpoint and their lambda callback functions + for (uint8_t i = 0; i < MAX_LIGHT_NUMBER; i++) { + pinMode(lightPins[i], OUTPUT); // set the GPIO function + OnOffLight[i].begin(false); // off + + // inline lambda function using capture array index -> it will just print a message in the console + OnOffLight[i].onChangeOnOff([i](bool state) -> bool { + // Display message with the specific light name and details + Serial.printf( + "Matter App Control: '%s' (OnOffLight[%d], Endpoint %d, GPIO %d) changed to: %s\r\n", lightName[i], i, OnOffLight[i].getEndPointId(), lightPins[i], + state ? "ON" : "OFF" + ); + + return true; + }); + } + // last step, starting Matter Stack + Matter.begin(); +} + +void loop() { + // Check Matter Plugin Commissioning state, which may change during execution of loop() + if (!Matter.isDeviceCommissioned()) { + Serial.println(""); + Serial.println("Matter Node is not commissioned yet."); + Serial.println("Initiate the device discovery in your Matter environment."); + Serial.println("Commission it to your Matter hub with the manual pairing code or QR code"); + Serial.printf("Manual pairing code: %s\r\n", Matter.getManualPairingCode().c_str()); + Serial.printf("QR code URL: %s\r\n", Matter.getOnboardingQRCodeUrl().c_str()); + // waits for Matter Plugin Commissioning. + uint32_t timeCount = 0; + while (!Matter.isDeviceCommissioned()) { + delay(100); + if ((timeCount++ % 50) == 0) { // 50*100ms = 5 sec + Serial.println("Matter Node not commissioned yet. Waiting for commissioning."); + } + } + } else { + if (Matter.isDeviceConnected()) { + Serial.println("Matter Node is commissioned and connected to the network. Ready for use."); + } else { + Serial.println("Matter Node is commissioned. Waiting for the network connection."); + } + // wait 3 seconds for the network connection + delay(3000); + } + + delay(100); +} diff --git a/libraries/Matter/examples/MatterLambdaSingleCallbackManyEPs/ci.json b/libraries/Matter/examples/MatterLambdaSingleCallbackManyEPs/ci.json new file mode 100644 index 00000000000..90b393f9156 --- /dev/null +++ b/libraries/Matter/examples/MatterLambdaSingleCallbackManyEPs/ci.json @@ -0,0 +1,6 @@ +{ + "fqbn_append": "PartitionScheme=huge_app", + "requires": [ + "CONFIG_ESP_MATTER_ENABLE_DATA_MODEL=y" + ] +} diff --git a/libraries/Matter/examples/MatterMinimum/ci.json b/libraries/Matter/examples/MatterMinimum/ci.json index 556a8a9ee6b..90b393f9156 100644 --- a/libraries/Matter/examples/MatterMinimum/ci.json +++ b/libraries/Matter/examples/MatterMinimum/ci.json @@ -1,7 +1,6 @@ { "fqbn_append": "PartitionScheme=huge_app", "requires": [ - "CONFIG_SOC_WIFI_SUPPORTED=y", "CONFIG_ESP_MATTER_ENABLE_DATA_MODEL=y" ] } diff --git a/libraries/Matter/examples/MatterOccupancySensor/ci.json b/libraries/Matter/examples/MatterOccupancySensor/ci.json index 556a8a9ee6b..90b393f9156 100644 --- a/libraries/Matter/examples/MatterOccupancySensor/ci.json +++ b/libraries/Matter/examples/MatterOccupancySensor/ci.json @@ -1,7 +1,6 @@ { "fqbn_append": "PartitionScheme=huge_app", "requires": [ - "CONFIG_SOC_WIFI_SUPPORTED=y", "CONFIG_ESP_MATTER_ENABLE_DATA_MODEL=y" ] } diff --git a/libraries/Matter/examples/MatterOnIdentify/ci.json b/libraries/Matter/examples/MatterOnIdentify/ci.json index 556a8a9ee6b..90b393f9156 100644 --- a/libraries/Matter/examples/MatterOnIdentify/ci.json +++ b/libraries/Matter/examples/MatterOnIdentify/ci.json @@ -1,7 +1,6 @@ { "fqbn_append": "PartitionScheme=huge_app", "requires": [ - "CONFIG_SOC_WIFI_SUPPORTED=y", "CONFIG_ESP_MATTER_ENABLE_DATA_MODEL=y" ] } diff --git a/libraries/Matter/examples/MatterOnOffLight/ci.json b/libraries/Matter/examples/MatterOnOffLight/ci.json index 556a8a9ee6b..90b393f9156 100644 --- a/libraries/Matter/examples/MatterOnOffLight/ci.json +++ b/libraries/Matter/examples/MatterOnOffLight/ci.json @@ -1,7 +1,6 @@ { "fqbn_append": "PartitionScheme=huge_app", "requires": [ - "CONFIG_SOC_WIFI_SUPPORTED=y", "CONFIG_ESP_MATTER_ENABLE_DATA_MODEL=y" ] } diff --git a/libraries/Matter/examples/MatterOnOffPlugin/ci.json b/libraries/Matter/examples/MatterOnOffPlugin/ci.json index 556a8a9ee6b..90b393f9156 100644 --- a/libraries/Matter/examples/MatterOnOffPlugin/ci.json +++ b/libraries/Matter/examples/MatterOnOffPlugin/ci.json @@ -1,7 +1,6 @@ { "fqbn_append": "PartitionScheme=huge_app", "requires": [ - "CONFIG_SOC_WIFI_SUPPORTED=y", "CONFIG_ESP_MATTER_ENABLE_DATA_MODEL=y" ] } diff --git a/libraries/Matter/examples/MatterPressureSensor/ci.json b/libraries/Matter/examples/MatterPressureSensor/ci.json index 556a8a9ee6b..90b393f9156 100644 --- a/libraries/Matter/examples/MatterPressureSensor/ci.json +++ b/libraries/Matter/examples/MatterPressureSensor/ci.json @@ -1,7 +1,6 @@ { "fqbn_append": "PartitionScheme=huge_app", "requires": [ - "CONFIG_SOC_WIFI_SUPPORTED=y", "CONFIG_ESP_MATTER_ENABLE_DATA_MODEL=y" ] } diff --git a/libraries/Matter/examples/MatterSmartButon/ci.json b/libraries/Matter/examples/MatterSmartButon/ci.json index 556a8a9ee6b..90b393f9156 100644 --- a/libraries/Matter/examples/MatterSmartButon/ci.json +++ b/libraries/Matter/examples/MatterSmartButon/ci.json @@ -1,7 +1,6 @@ { "fqbn_append": "PartitionScheme=huge_app", "requires": [ - "CONFIG_SOC_WIFI_SUPPORTED=y", "CONFIG_ESP_MATTER_ENABLE_DATA_MODEL=y" ] } diff --git a/libraries/Matter/examples/MatterTemperatureLight/ci.json b/libraries/Matter/examples/MatterTemperatureLight/ci.json index 556a8a9ee6b..90b393f9156 100644 --- a/libraries/Matter/examples/MatterTemperatureLight/ci.json +++ b/libraries/Matter/examples/MatterTemperatureLight/ci.json @@ -1,7 +1,6 @@ { "fqbn_append": "PartitionScheme=huge_app", "requires": [ - "CONFIG_SOC_WIFI_SUPPORTED=y", "CONFIG_ESP_MATTER_ENABLE_DATA_MODEL=y" ] } diff --git a/libraries/Matter/examples/MatterTemperatureSensor/ci.json b/libraries/Matter/examples/MatterTemperatureSensor/ci.json index 556a8a9ee6b..90b393f9156 100644 --- a/libraries/Matter/examples/MatterTemperatureSensor/ci.json +++ b/libraries/Matter/examples/MatterTemperatureSensor/ci.json @@ -1,7 +1,6 @@ { "fqbn_append": "PartitionScheme=huge_app", "requires": [ - "CONFIG_SOC_WIFI_SUPPORTED=y", "CONFIG_ESP_MATTER_ENABLE_DATA_MODEL=y" ] } diff --git a/libraries/Matter/examples/MatterThermostat/ci.json b/libraries/Matter/examples/MatterThermostat/ci.json index 556a8a9ee6b..90b393f9156 100644 --- a/libraries/Matter/examples/MatterThermostat/ci.json +++ b/libraries/Matter/examples/MatterThermostat/ci.json @@ -1,7 +1,6 @@ { "fqbn_append": "PartitionScheme=huge_app", "requires": [ - "CONFIG_SOC_WIFI_SUPPORTED=y", "CONFIG_ESP_MATTER_ENABLE_DATA_MODEL=y" ] } diff --git a/libraries/Matter/examples/WiFiProvWithinMatter/WiFiProvWithinMatter.ino b/libraries/Matter/examples/WiFiProvWithinMatter/WiFiProvWithinMatter.ino deleted file mode 100644 index 3434217624d..00000000000 --- a/libraries/Matter/examples/WiFiProvWithinMatter/WiFiProvWithinMatter.ino +++ /dev/null @@ -1,152 +0,0 @@ -/* - Please read README.md file in this folder, or on the web: - https://github.com/espressif/arduino-esp32/tree/master/libraries/WiFiProv/examples/WiFiProv - - Note: This sketch takes up a lot of space for the app and may not be able to flash with default setting on some chips. - If you see Error like this: "Sketch too big" - In Arduino IDE go to: Tools > Partition scheme > chose anything that has more than 1.4MB APP - - for example "No OTA (2MB APP/2MB SPIFFS)" - - This example demonstrates that it is possible to provision WiFi using BLE or Software AP using - the ESP BLE Prov APP or ESP SoftAP Provisioning APP from Android Play or/and iOS APP Store - - Once the WiFi is provisioned, Matter will start its process as usual. - - This same Example could be used for any other WiFi Provisioning method. -*/ - -// Matter Manager -#include -#include -#include - -#if !CONFIG_BLUEDROID_ENABLED -#define USE_SOFT_AP // ESP32-S2 has no BLE, therefore, it shall use SoftAP Provisioning -#endif -//#define USE_SOFT_AP // Uncomment if you want to enforce using the Soft AP method instead of BLE - -const char *pop = "abcd1234"; // Proof of possession - otherwise called a PIN - string provided by the device, entered by the user in the phone app -const char *service_name = "PROV_123"; // Name of your device (the Espressif apps expects by default device name starting with "Prov_") -const char *service_key = NULL; // Password used for SofAP method (NULL = no password needed) -bool reset_provisioned = true; // When true the library will automatically delete previously provisioned data. - -// List of Matter Endpoints for this Node -// Single On/Off Light Endpoint - at least one per node -MatterOnOffLight OnOffLight; - -// Light GPIO that can be controlled by Matter APP -#ifdef LED_BUILTIN -const uint8_t ledPin = LED_BUILTIN; -#else -const uint8_t ledPin = 2; // Set your pin here if your board has not defined LED_BUILTIN -#endif - -// set your board USER BUTTON pin here - decommissioning button -const uint8_t buttonPin = BOOT_PIN; // Set your pin here. Using BOOT Button. - -// Button control - decommision the Matter Node -uint32_t button_time_stamp = 0; // debouncing control -bool button_state = false; // false = released | true = pressed -const uint32_t decommissioningTimeout = 5000; // keep the button pressed for 5s, or longer, to decommission - -// Matter Protocol Endpoint (On/OFF Light) Callback -bool matterCB(bool state) { - digitalWrite(ledPin, state ? HIGH : LOW); - // This callback must return the success state to Matter core - return true; -} - -void setup() { - // Initialize the USER BUTTON (Boot button) that will be used to decommission the Matter Node - pinMode(buttonPin, INPUT_PULLUP); - - Serial.begin(115200); - // Initialize the LED GPIO - pinMode(ledPin, OUTPUT); - - WiFi.begin(); // no SSID/PWD - get it from the Provisioning APP or from NVS (last successful connection) - - // BLE Provisioning using the ESP SoftAP Prov works fine for any BLE SoC, including ESP32, ESP32S3 and ESP32C3. -#if CONFIG_BLUEDROID_ENABLED && !defined(USE_SOFT_AP) - Serial.println("Begin Provisioning using BLE"); - // Sample uuid that user can pass during provisioning using BLE - uint8_t uuid[16] = {0xb4, 0xdf, 0x5a, 0x1c, 0x3f, 0x6b, 0xf4, 0xbf, 0xea, 0x4a, 0x82, 0x03, 0x04, 0x90, 0x1a, 0x02}; - WiFiProv.beginProvision( - NETWORK_PROV_SCHEME_BLE, NETWORK_PROV_SCHEME_HANDLER_FREE_BLE, NETWORK_PROV_SECURITY_1, pop, service_name, service_key, uuid, reset_provisioned - ); - Serial.println("You may use this BLE QRCode:"); - WiFiProv.printQR(service_name, pop, "ble"); -#else - Serial.println("Begin Provisioning using Soft AP"); - WiFiProv.beginProvision(NETWORK_PROV_SCHEME_SOFTAP, NETWORK_PROV_SCHEME_HANDLER_NONE, NETWORK_PROV_SECURITY_1, pop, service_name, service_key); - Serial.println("You may use this WiFi QRCode:"); - WiFiProv.printQR(service_name, pop, "softap"); -#endif - - // Wait for WiFi connection - uint32_t counter = 0; - while (WiFi.status() != WL_CONNECTED) { - // resets the device after 10 minutes - if (counter > 2 * 60 * 10) { - Serial.println("\r\n================================================"); - Serial.println("Already 10 minutes past. The device will reboot."); - Serial.println("================================================\r\n"); - Serial.flush(); // wait until the Serial has sent the whole message. - ESP.restart(); - } - // WiFi searching feedback - Serial.print("."); - delay(500); - // adds a new line every 30 seconds - counter++; - if (!(counter % 60)) { - Serial.println(); - } - } - - // WiFi shall be connected by now - Serial.println(); - - // Initialize at least one Matter EndPoint - OnOffLight.begin(); - - // Associate a callback to the Matter Controller - OnOffLight.onChange(matterCB); - - // Matter beginning - Last step, after all EndPoints are initialized - Matter.begin(); - - while (!Matter.isDeviceCommissioned()) { - Serial.println("Matter Node is not commissioned yet."); - Serial.println("Initiate the device discovery in your Matter environment."); - Serial.println("Commission it to your Matter hub with the manual pairing code or QR code"); - Serial.printf("Manual pairing code: %s\r\n", Matter.getManualPairingCode().c_str()); - Serial.printf("QR code URL: %s\r\n", Matter.getOnboardingQRCodeUrl().c_str()); - Serial.println(); - // waits 30 seconds for Matter Commissioning, keeping it blocked until done - delay(30000); - } -} - -void loop() { - // Check if the button has been pressed - if (digitalRead(buttonPin) == LOW && !button_state) { - // deals with button debouncing - button_time_stamp = millis(); // record the time while the button is pressed. - button_state = true; // pressed. - } - - if (digitalRead(buttonPin) == HIGH && button_state) { - button_state = false; // released - } - - // Onboard User Button is kept pressed for longer than 5 seconds in order to decommission matter node - uint32_t time_diff = millis() - button_time_stamp; - if (button_state && time_diff > decommissioningTimeout) { - Serial.println("Decommissioning the Light Matter Accessory. It shall be commissioned again."); - Matter.decommission(); - button_time_stamp = millis(); // avoid running decommissining again, reboot takes a second or so - } - - delay(500); -} diff --git a/libraries/Matter/examples/WiFiProvWithinMatter/ci.json b/libraries/Matter/examples/WiFiProvWithinMatter/ci.json deleted file mode 100644 index 0665800b12b..00000000000 --- a/libraries/Matter/examples/WiFiProvWithinMatter/ci.json +++ /dev/null @@ -1,7 +0,0 @@ -{ - "fqbn_append": "PartitionScheme=huge_app", - "requires": [ - "CONFIG_SOC_WIFI_SUPPORTED=y", - "CONFIG_ESP_MATTER_ENABLE_DATA_MODEL=y" - ] -} diff --git a/libraries/Matter/keywords.txt b/libraries/Matter/keywords.txt index 68aaebb1d4d..6c2e092e417 100644 --- a/libraries/Matter/keywords.txt +++ b/libraries/Matter/keywords.txt @@ -36,8 +36,10 @@ EndPointSpeedCB KEYWORD1 EndPointOnOffCB KEYWORD1 EndPointBrightnessCB KEYWORD1 EndPointRGBColorCB KEYWORD1 +EndPointIdentifyCB KEYWORD1 matterEvent_t KEYWORD1 matterEventCB KEYWORD1 +attrOperation_t KEYWORD1 ####################################### # Methods and Functions (KEYWORD2) @@ -111,6 +113,12 @@ onChangeLocalTemperature KEYWORD2 onChangeCoolingSetpoint KEYWORD2 onChangeHeatingSetpoint KEYWORD2 onEvent KEYWORD2 +setEndPointId KEYWORD2 +getEndPointId KEYWORD2 +getSecondaryNetworkEndPointId KEYWORD2 +createSecondaryNetworkInterface KEYWORD2 +onIdentify KEYWORD2 +endpointIdentifyCB KEYWORD2 ####################################### # Constants (LITERAL1) From 0a45a0614244002f82fe7f006effeda7c8075469 Mon Sep 17 00:00:00 2001 From: Sugar Glider Date: Sat, 12 Jul 2025 02:12:36 -0300 Subject: [PATCH 108/173] feat(wire): std::functional Wire slave callback functions (#11582) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit This PR enhances the Wire library to support std::function–based callbacks for I2C slave mode, enabling the use of lambdas and captured contexts. - Replaces raw function pointers in TwoWire and HardwareI2C with std::function for onRequest and onReceive - Updates constructors, method signatures, and default initializations to use std::function - Adds new example sketch, CI config, and documentation updates demonstrating the functional callback API --- cores/esp32/HardwareI2C.h | 6 +- docs/en/api/i2c.rst | 135 +++++++++++++++++- .../WireSlaveFunctionalCallback.ino | 37 +++++ .../WireSlaveFunctionalCallback/ci.json | 5 + libraries/Wire/src/Wire.cpp | 6 +- libraries/Wire/src/Wire.h | 12 +- 6 files changed, 184 insertions(+), 17 deletions(-) create mode 100644 libraries/Wire/examples/WireSlaveFunctionalCallback/WireSlaveFunctionalCallback.ino create mode 100644 libraries/Wire/examples/WireSlaveFunctionalCallback/ci.json diff --git a/cores/esp32/HardwareI2C.h b/cores/esp32/HardwareI2C.h index 65b7e2036b2..c44f34e1ee7 100644 --- a/cores/esp32/HardwareI2C.h +++ b/cores/esp32/HardwareI2C.h @@ -20,6 +20,7 @@ #include #include "Stream.h" +#include class HardwareI2C : public Stream { public: @@ -36,6 +37,7 @@ class HardwareI2C : public Stream { virtual size_t requestFrom(uint8_t address, size_t len, bool stopBit) = 0; virtual size_t requestFrom(uint8_t address, size_t len) = 0; - virtual void onReceive(void (*)(int)) = 0; - virtual void onRequest(void (*)(void)) = 0; + // Update base class to use std::function + virtual void onReceive(const std::function &) = 0; + virtual void onRequest(const std::function &) = 0; }; diff --git a/docs/en/api/i2c.rst b/docs/en/api/i2c.rst index eac04b76a23..06d4d1953a6 100644 --- a/docs/en/api/i2c.rst +++ b/docs/en/api/i2c.rst @@ -347,20 +347,147 @@ This function will return ``true`` if the peripheral was initialized correctly. onReceive ^^^^^^^^^ -The ``onReceive`` function is used to define the callback for the data received from the master. +The ``onReceive`` function is used to define the callback for data received from the master device. .. code-block:: arduino - void onReceive( void (*)(int) ); + void onReceive(const std::function& callback); + +**Function Signature:** + +The callback function must have the signature ``void(int numBytes)`` where ``numBytes`` indicates how many bytes were received from the master. + +**Usage Examples:** + +.. code-block:: arduino + + // Method 1: Regular function + void handleReceive(int numBytes) { + Serial.printf("Received %d bytes: ", numBytes); + while (Wire.available()) { + char c = Wire.read(); + Serial.print(c); + } + Serial.println(); + } + Wire.onReceive(handleReceive); + + // Method 2: Lambda function + Wire.onReceive([](int numBytes) { + Serial.printf("Master sent %d bytes\n", numBytes); + while (Wire.available()) { + uint8_t data = Wire.read(); + // Process received data + Serial.printf("Data: 0x%02X\n", data); + } + }); + + // Method 3: Lambda with capture (for accessing variables) + int deviceId = 42; + Wire.onReceive([deviceId](int numBytes) { + Serial.printf("Device %d received %d bytes\n", deviceId, numBytes); + // Process data... + }); + + // Method 4: Using std::function variable + std::function receiveHandler = [](int bytes) { + Serial.printf("Handling %d received bytes\n", bytes); + }; + Wire.onReceive(receiveHandler); + + // Method 5: Class member function (using lambda wrapper) + class I2CDevice { + private: + int deviceAddress; + public: + I2CDevice(int addr) : deviceAddress(addr) {} + + void handleReceive(int numBytes) { + Serial.printf("Device 0x%02X received %d bytes\n", deviceAddress, numBytes); + } + + void setup() { + Wire.onReceive([this](int bytes) { + this->handleReceive(bytes); + }); + } + }; + +.. note:: + The ``onReceive`` callback is triggered when the I2C master sends data to this slave device. + Use ``Wire.available()`` and ``Wire.read()`` inside the callback to retrieve the received data. onRequest ^^^^^^^^^ -The ``onRequest`` function is used to define the callback for the data to be send to the master. +The ``onRequest`` function is used to define the callback for responding to master read requests. + +.. code-block:: arduino + + void onRequest(const std::function& callback); + +**Function Signature:** + +The callback function must have the signature ``void()`` with no parameters. This callback is triggered when the master requests data from this slave device. + +**Usage Examples:** .. code-block:: arduino - void onRequest( void (*)(void) ); + // Method 1: Regular function + void handleRequest() { + static int counter = 0; + Wire.printf("Response #%d", counter++); + } + Wire.onRequest(handleRequest); + + // Method 2: Lambda function + Wire.onRequest([]() { + // Send sensor data to master + int sensorValue = analogRead(A0); + Wire.write(sensorValue >> 8); // High byte + Wire.write(sensorValue & 0xFF); // Low byte + }); + + // Method 3: Lambda with capture (for accessing variables) + int deviceStatus = 1; + String deviceName = "Sensor1"; + Wire.onRequest([&deviceStatus, &deviceName]() { + Wire.write(deviceStatus); + Wire.write(deviceName.c_str(), deviceName.length()); + }); + + // Method 4: Using std::function variable + std::function requestHandler = []() { + Wire.write("Hello Master!"); + }; + Wire.onRequest(requestHandler); + + // Method 5: Class member function (using lambda wrapper) + class TemperatureSensor { + private: + float temperature; + public: + void updateTemperature() { + temperature = 25.5; // Read from actual sensor + } + + void sendTemperature() { + // Convert float to bytes and send + uint8_t* tempBytes = (uint8_t*)&temperature; + Wire.write(tempBytes, sizeof(float)); + } + + void setup() { + Wire.onRequest([this]() { + this->sendTemperature(); + }); + } + }; + +.. note:: + The ``onRequest`` callback is triggered when the I2C master requests data from this slave device. + Use ``Wire.write()`` inside the callback to send response data back to the master. slaveWrite ^^^^^^^^^^ diff --git a/libraries/Wire/examples/WireSlaveFunctionalCallback/WireSlaveFunctionalCallback.ino b/libraries/Wire/examples/WireSlaveFunctionalCallback/WireSlaveFunctionalCallback.ino new file mode 100644 index 00000000000..a18fd2f023e --- /dev/null +++ b/libraries/Wire/examples/WireSlaveFunctionalCallback/WireSlaveFunctionalCallback.ino @@ -0,0 +1,37 @@ +// This example demonstrates the use of functional callbacks with the Wire library +// for I2C slave communication. It shows how to handle requests and data reception + +#include "Wire.h" + +#define I2C_DEV_ADDR 0x55 + +uint32_t i = 0; + +void setup() { + Serial.begin(115200); + Serial.setDebugOutput(true); + + Wire.onRequest([]() { + Wire.print(i++); + Wire.print(" Packets."); + Serial.println("onRequest"); + }); + + Wire.onReceive([](int len) { + Serial.printf("onReceive[%d]: ", len); + while (Wire.available()) { + Serial.write(Wire.read()); + } + Serial.println(); + }); + + Wire.begin((uint8_t)I2C_DEV_ADDR); + +#if CONFIG_IDF_TARGET_ESP32 + char message[64]; + snprintf(message, 64, "%lu Packets.", i++); + Wire.slaveWrite((uint8_t *)message, strlen(message)); +#endif +} + +void loop() {} diff --git a/libraries/Wire/examples/WireSlaveFunctionalCallback/ci.json b/libraries/Wire/examples/WireSlaveFunctionalCallback/ci.json new file mode 100644 index 00000000000..3c877975d62 --- /dev/null +++ b/libraries/Wire/examples/WireSlaveFunctionalCallback/ci.json @@ -0,0 +1,5 @@ +{ + "requires": [ + "CONFIG_SOC_I2C_SUPPORT_SLAVE=y" + ] +} diff --git a/libraries/Wire/src/Wire.cpp b/libraries/Wire/src/Wire.cpp index 34c814b5117..cda098d2d5b 100644 --- a/libraries/Wire/src/Wire.cpp +++ b/libraries/Wire/src/Wire.cpp @@ -48,7 +48,7 @@ TwoWire::TwoWire(uint8_t bus_num) #endif #if SOC_I2C_SUPPORT_SLAVE , - is_slave(false), user_onRequest(NULL), user_onReceive(NULL) + is_slave(false), user_onRequest(nullptr), user_onReceive(nullptr) #endif /* SOC_I2C_SUPPORT_SLAVE */ { } @@ -596,14 +596,14 @@ void TwoWire::flush() { //i2cFlush(num); // cleanup } -void TwoWire::onReceive(void (*function)(int)) { +void TwoWire::onReceive(const std::function &function) { #if SOC_I2C_SUPPORT_SLAVE user_onReceive = function; #endif } // sets function called on slave read -void TwoWire::onRequest(void (*function)(void)) { +void TwoWire::onRequest(const std::function &function) { #if SOC_I2C_SUPPORT_SLAVE user_onRequest = function; #endif diff --git a/libraries/Wire/src/Wire.h b/libraries/Wire/src/Wire.h index b84aa5b2131..9cebdfaa304 100644 --- a/libraries/Wire/src/Wire.h +++ b/libraries/Wire/src/Wire.h @@ -48,10 +48,6 @@ #ifndef I2C_BUFFER_LENGTH #define I2C_BUFFER_LENGTH 128 // Default size, if none is set using Wire::setBuffersize(size_t) #endif -#if SOC_I2C_SUPPORT_SLAVE -typedef void (*user_onRequest)(void); -typedef void (*user_onReceive)(uint8_t *, int); -#endif /* SOC_I2C_SUPPORT_SLAVE */ class TwoWire : public HardwareI2C { protected: @@ -77,8 +73,8 @@ class TwoWire : public HardwareI2C { private: #if SOC_I2C_SUPPORT_SLAVE bool is_slave; - void (*user_onRequest)(void); - void (*user_onReceive)(int); + std::function user_onRequest; + std::function user_onReceive; static void onRequestService(uint8_t, void *); static void onReceiveService(uint8_t, uint8_t *, size_t, bool, void *); #endif /* SOC_I2C_SUPPORT_SLAVE */ @@ -116,8 +112,8 @@ class TwoWire : public HardwareI2C { size_t requestFrom(uint8_t address, size_t len, bool stopBit) override; size_t requestFrom(uint8_t address, size_t len) override; - void onReceive(void (*)(int)) override; - void onRequest(void (*)(void)) override; + void onReceive(const std::function &) override; + void onRequest(const std::function &) override; //call setPins() first, so that begin() can be called without arguments from libraries bool setPins(int sda, int scl); From 1f0d4b5dc0d6008583ac9527907325b9f2506caa Mon Sep 17 00:00:00 2001 From: Lucas Saavedra Vaz <32426024+lucasssvaz@users.noreply.github.com> Date: Mon, 14 Jul 2025 13:17:56 -0300 Subject: [PATCH 109/173] ci(gitlab): Initial GitLab setup (#11577) * ci(gitlab): Initial GitLab setup * fix(version): Add to version update script * ci(pre-commit): Apply automatic fixes --------- Co-authored-by: pre-commit-ci-lite[bot] <117423508+pre-commit-ci-lite[bot]@users.noreply.github.com> --- .github/CODEOWNERS | 1 + .github/scripts/update-version.sh | 5 +++++ .gitlab-ci.yml | 25 +++++++++++++++++++++++++ .gitlab/workflows/common.yml | 26 ++++++++++++++++++++++++++ .gitlab/workflows/sample.yml | 6 ++++++ 5 files changed, 63 insertions(+) create mode 100644 .gitlab-ci.yml create mode 100644 .gitlab/workflows/common.yml create mode 100644 .gitlab/workflows/sample.yml diff --git a/.github/CODEOWNERS b/.github/CODEOWNERS index 75a2b46d619..64d241ba20a 100644 --- a/.github/CODEOWNERS +++ b/.github/CODEOWNERS @@ -11,6 +11,7 @@ # CI /.github/ @lucasssvaz @me-no-dev @P-R-O-C-H-Y +/.gitlab/ @lucasssvaz /tests/ @lucasssvaz @P-R-O-C-H-Y # Tools diff --git a/.github/scripts/update-version.sh b/.github/scripts/update-version.sh index 622f2fe8ff8..59a95d01105 100755 --- a/.github/scripts/update-version.sh +++ b/.github/scripts/update-version.sh @@ -45,6 +45,11 @@ cat docs/conf_common.py | \ sed "s/.. |version| replace:: .*/.. |version| replace:: $ESP_ARDUINO_VERSION/g" | \ sed "s/.. |idf_version| replace:: .*/.. |idf_version| replace:: $ESP_IDF_VERSION/g" > docs/__conf_common.py && mv docs/__conf_common.py docs/conf_common.py +echo "Updating .gitlab/workflows/common.yml..." +cat .gitlab/workflows/common.yml | \ +sed "s/ESP_IDF_VERSION:.*/ESP_IDF_VERSION: \"$ESP_IDF_VERSION\"/g" | \ +sed "s/ESP_ARDUINO_VERSION:.*/ESP_ARDUINO_VERSION: \"$ESP_ARDUINO_VERSION\"/g" > .gitlab/workflows/__common.yml && mv .gitlab/workflows/__common.yml .gitlab/workflows/common.yml + echo "Updating cores/esp32/esp_arduino_version.h..." cat cores/esp32/esp_arduino_version.h | \ sed "s/#define ESP_ARDUINO_VERSION_MAJOR.*/#define ESP_ARDUINO_VERSION_MAJOR $ESP_ARDUINO_VERSION_MAJOR/g" | \ diff --git a/.gitlab-ci.yml b/.gitlab-ci.yml new file mode 100644 index 00000000000..89a45022bc2 --- /dev/null +++ b/.gitlab-ci.yml @@ -0,0 +1,25 @@ +workflow: + rules: + # Disable those non-protected push triggered pipelines + - if: '$CI_COMMIT_REF_NAME != "master" && $CI_COMMIT_BRANCH !~ /^release\/v/ && $CI_COMMIT_TAG !~ /^\d+\.\d+(\.\d+)?($|-)/ && $CI_PIPELINE_SOURCE == "push"' + when: never + # when running merged result pipelines, CI_COMMIT_SHA represents the temp commit it created. + # Please use PIPELINE_COMMIT_SHA at all places that require a commit sha of the original commit. + - if: $CI_OPEN_MERGE_REQUESTS != null + variables: + PIPELINE_COMMIT_SHA: $CI_MERGE_REQUEST_SOURCE_BRANCH_SHA + IS_MR_PIPELINE: 1 + - if: $CI_OPEN_MERGE_REQUESTS == null + variables: + PIPELINE_COMMIT_SHA: $CI_COMMIT_SHA + IS_MR_PIPELINE: 0 + - if: '$CI_PIPELINE_SOURCE == "schedule"' + variables: + IS_SCHEDULED_RUN: "true" + - when: always + +# Place the default settings in `.gitlab/workflows/common.yml` instead + +include: + - ".gitlab/workflows/common.yml" + - ".gitlab/workflows/sample.yml" diff --git a/.gitlab/workflows/common.yml b/.gitlab/workflows/common.yml new file mode 100644 index 00000000000..9086da018ab --- /dev/null +++ b/.gitlab/workflows/common.yml @@ -0,0 +1,26 @@ +##################### +# Default Variables # +##################### + +stages: + - pre_check + - build + - test + - result + +variables: + ESP_IDF_VERSION: "5.4" + ESP_ARDUINO_VERSION: "3.2.1" + +############# +# `default` # +############# + +default: + retry: + max: 2 + when: + # In case of a runner failure we could hop to another one, or a network error could go away. + - runner_system_failure + # Job execution timeout may be caused by a network issue. + - job_execution_timeout diff --git a/.gitlab/workflows/sample.yml b/.gitlab/workflows/sample.yml new file mode 100644 index 00000000000..32b6fce042d --- /dev/null +++ b/.gitlab/workflows/sample.yml @@ -0,0 +1,6 @@ +hello-world: + stage: test + script: + - echo "Hello, World from GitLab CI!" + rules: + - if: $CI_PIPELINE_SOURCE == "push" From 82d56bc679dfcd8de2fb06596ff4c4bcca8e6247 Mon Sep 17 00:00:00 2001 From: Sugar Glider Date: Wed, 16 Jul 2025 05:51:31 -0300 Subject: [PATCH 110/173] feat(gpio): new functional interrupt lambda example (#11589) * feat(gpio): new functional interrupt lambda example * fix(readme): schematic diagram allignment * fix(example): uses volatile for ISR variables Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> * fix(example): uses volatile for ISR variables Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> * fix(example): uses volatile data type also for the pointers * fix(readme): clear documentation * feat(example): improves ISR execution time Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> * feat(gpio): simplifies the example and documentation * feat(gpio): uses IRAM lambda and fixes volatile operation * fix(doc): fixing documentation apresentation * ci(pre-commit): Apply automatic fixes * fix(ci): Update README.md --------- Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> Co-authored-by: pre-commit-ci-lite[bot] <117423508+pre-commit-ci-lite[bot]@users.noreply.github.com> Co-authored-by: Me No Dev --- .../FunctionalInterruptLambda.ino | 157 ++++++++++++++++++ .../GPIO/FunctionalInterruptLambda/README.md | 147 ++++++++++++++++ 2 files changed, 304 insertions(+) create mode 100644 libraries/ESP32/examples/GPIO/FunctionalInterruptLambda/FunctionalInterruptLambda.ino create mode 100644 libraries/ESP32/examples/GPIO/FunctionalInterruptLambda/README.md diff --git a/libraries/ESP32/examples/GPIO/FunctionalInterruptLambda/FunctionalInterruptLambda.ino b/libraries/ESP32/examples/GPIO/FunctionalInterruptLambda/FunctionalInterruptLambda.ino new file mode 100644 index 00000000000..57d35383f17 --- /dev/null +++ b/libraries/ESP32/examples/GPIO/FunctionalInterruptLambda/FunctionalInterruptLambda.ino @@ -0,0 +1,157 @@ +/* + SPDX-FileCopyrightText: 2025 Espressif Systems (Shanghai) CO LTD + + SPDX-License-Identifier: Apache-2.0 + + 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. + + ESP32 Lambda FunctionalInterrupt Example + ======================================== + + This example demonstrates how to use lambda functions with FunctionalInterrupt + for GPIO pin interrupt callbacks on ESP32. It shows CHANGE mode detection + with LED toggle functionality and proper debouncing. + + Hardware Setup: + - Use BOOT Button or connect a button between BUTTON_PIN and GND (with internal pullup) + - Use Builtin Board LED or connect an LED with resistor to GPIO 2 (LED_PIN) + + Features Demonstrated: + 1. CHANGE mode lambda to detect both RISING and FALLING edges + 2. LED toggle on button press (FALLING edge) + 3. Edge type detection using digitalRead() within ISR + 4. Hardware debouncing with configurable timeout + + IMPORTANT NOTE ABOUT ESP32 INTERRUPT BEHAVIOR: + - Only ONE interrupt handler can be attached per GPIO pin at a time + - Calling attachInterrupt() on a pin that already has an interrupt will override the previous one + - This applies regardless of edge type (RISING, FALLING, CHANGE) + - If you need both RISING and FALLING detection on the same pin, use CHANGE mode + and determine the edge type within your handler by reading the pin state +*/ + +#include +#include + +// Pin definitions +#define BUTTON_PIN BOOT_PIN // BOOT BUTTON - change as needed +#ifdef LED_BUILTIN +#define LED_PIN LED_BUILTIN +#else +#warning Using LED_PIN = GPIO 2 as default - change as needed +#define LED_PIN 2 // change as needed +#endif + +// Global variables for interrupt handling (volatile for ISR safety) +volatile uint32_t buttonPressCount = 0; +volatile uint32_t buttonReleaseCount = 0; +volatile bool buttonPressed = false; +volatile bool buttonReleased = false; +volatile bool ledState = false; +volatile bool ledStateChanged = false; // Flag to indicate LED needs updating + +// Debouncing variables (volatile for ISR safety) +volatile unsigned long lastButtonInterruptTime = 0; +const unsigned long DEBOUNCE_DELAY_MS = 50; // 50ms debounce delay + +// State-based debouncing to prevent hysteresis issues +volatile bool lastButtonState = HIGH; // Track last stable state (HIGH = released) + +// Global lambda function (declared at file scope) - ISR in IRAM +IRAM_ATTR std::function changeModeLambda = []() { + // Simple debouncing: check if enough time has passed since last interrupt + unsigned long currentTime = millis(); + if (currentTime - lastButtonInterruptTime < DEBOUNCE_DELAY_MS) { + return; // Ignore this interrupt due to bouncing + } + + // Read current pin state to determine edge type + bool currentState = digitalRead(BUTTON_PIN); + + // State-based debouncing: only process if state actually changed + if (currentState == lastButtonState) { + return; // No real state change, ignore (hysteresis/noise) + } + + // Update timing and state + lastButtonInterruptTime = currentTime; + lastButtonState = currentState; + + if (currentState == LOW) { + // FALLING edge detected (button pressed) - set flag for main loop + // volatile variables require use of temporary value transfer + uint32_t temp = buttonPressCount + 1; + buttonPressCount = temp; + buttonPressed = true; + ledStateChanged = true; // Signal main loop to toggle LED + } else { + // RISING edge detected (button released) - set flag for main loop + // volatile variables require use of temporary value transfer + uint32_t temp = buttonReleaseCount + 1; + buttonReleaseCount = temp; + buttonReleased = true; + } +}; + +void setup() { + Serial.begin(115200); + delay(1000); // Allow serial monitor to connect + + Serial.println("ESP32 Lambda FunctionalInterrupt Example"); + Serial.println("========================================"); + + // Configure pins + pinMode(BUTTON_PIN, INPUT_PULLUP); + pinMode(LED_PIN, OUTPUT); + digitalWrite(LED_PIN, LOW); + + // CHANGE mode lambda to handle both RISING and FALLING edges + // This toggles the LED on button press (FALLING edge) + Serial.println("Setting up CHANGE mode lambda for LED toggle"); + + // Use the global lambda function + attachInterrupt(BUTTON_PIN, changeModeLambda, CHANGE); + + Serial.println(); + Serial.printf("Lambda interrupt configured on Pin %d (CHANGE mode)\r\n", BUTTON_PIN); + Serial.printf("Debounce delay: %lu ms\r\n", DEBOUNCE_DELAY_MS); + Serial.println(); + Serial.println("Press the button to toggle the LED!"); + Serial.println("Button press (FALLING edge) will toggle the LED."); + Serial.println("Button release (RISING edge) will be detected and reported."); + Serial.println("Button includes debouncing to prevent mechanical bounce issues."); + Serial.println(); +} + +void loop() { + // Handle LED state changes (ISR-safe approach) + if (ledStateChanged) { + ledStateChanged = false; + ledState = !ledState; // Toggle LED state in main loop + digitalWrite(LED_PIN, ledState); + } + + // Check for button presses + if (buttonPressed) { + buttonPressed = false; + Serial.printf("==> Button PRESSED! Count: %lu, LED: %s (FALLING edge)\r\n", buttonPressCount, ledState ? "ON" : "OFF"); + } + + // Check for button releases + if (buttonReleased) { + buttonReleased = false; + Serial.printf("==> Button RELEASED! Count: %lu (RISING edge)\r\n", buttonReleaseCount); + } + + delay(10); +} diff --git a/libraries/ESP32/examples/GPIO/FunctionalInterruptLambda/README.md b/libraries/ESP32/examples/GPIO/FunctionalInterruptLambda/README.md new file mode 100644 index 00000000000..9488c317fde --- /dev/null +++ b/libraries/ESP32/examples/GPIO/FunctionalInterruptLambda/README.md @@ -0,0 +1,147 @@ +# ESP32 Lambda FunctionalInterrupt Example + +This example demonstrates how to use lambda functions with FunctionalInterrupt for GPIO pin interrupt callbacks on ESP32. It shows CHANGE mode detection with LED toggle functionality and proper debouncing. + +## Features Demonstrated + +1. **CHANGE mode lambda** to detect both RISING and FALLING edges +2. **LED toggle on button press** (FALLING edge) +3. **Edge type detection** using digitalRead() within ISR +4. **Hardware debouncing** with configurable timeout +5. **IRAM_ATTR lambda declaration** for optimal ISR performance in RAM + +## Hardware Setup + +- Use BOOT Button or connect a button between BUTTON_PIN and GND (with internal pullup) +- Use Builtin Board LED (no special hardware setup) or connect an LED with resistor to GPIO assigned as LED_PIN.\ + Some boards have an RGB LED that needs no special hardware setup to work as a simple white on/off LED. + +``` +ESP32 Board Button/LED +----------- --------- +BOOT_PIN ------------ [BUTTON] ---- GND +LED_PIN --------------- [LED] ----- GND + ¦ + [330O] (*) Only needed when using an external LED attached to the GPIO. + ¦ + 3V3 +``` + +## Important ESP32 Interrupt Behavior + +**CRITICAL:** Only ONE interrupt handler can be attached per GPIO pin at a time on ESP32. + +- Calling `attachInterrupt()` on a pin that already has an interrupt will **override** the previous one +- This applies regardless of edge type (RISING, FALLING, CHANGE) +- If you need both RISING and FALLING detection on the same pin, use **CHANGE mode** and determine the edge type within your handler by reading the pin state + +## Code Overview + +This example demonstrates a simple CHANGE mode lambda interrupt that: + +- **Detects both button press and release** using a single interrupt handler +- **Toggles LED only on button press** (FALLING edge) +- **Reports both press and release events** to Serial output +- **Uses proper debouncing** to prevent switch bounce issues +- **Implements minimal lambda captures** for simplicity + +## Lambda Function Pattern + +### CHANGE Mode Lambda with IRAM Declaration +```cpp +// Global lambda declared with IRAM_ATTR for optimal ISR performance +IRAM_ATTR std::function changeModeLambda = []() { + // Debouncing check + unsigned long currentTime = millis(); + if (currentTime - lastButtonInterruptTime < DEBOUNCE_DELAY_MS) { + return; // Ignore bouncing + } + + // Determine edge type + bool currentState = digitalRead(BUTTON_PIN); + if (currentState == lastButtonState) { + return; // No real state change + } + + // Update state and handle edges + lastButtonInterruptTime = currentTime; + lastButtonState = currentState; + + if (currentState == LOW) { + // Button pressed (FALLING edge) + buttonPressCount++; + buttonPressed = true; + ledStateChanged = true; // Signal LED toggle + } else { + // Button released (RISING edge) + buttonReleaseCount++; + buttonReleased = true; + } +}; + +attachInterrupt(BUTTON_PIN, changeModeLambda, CHANGE); +``` + +## Key Concepts + +### Edge Detection in CHANGE Mode +```cpp +if (digitalRead(pin) == LOW) { + // FALLING edge detected (button pressed) +} else { + // RISING edge detected (button released) +} +``` + +### Debouncing Strategy +This example implements dual-layer debouncing: +1. **Time-based**: Ignores interrupts within 50 ms of previous one +2. **State-based**: Only processes actual state changes + +### Main Loop Processing +```cpp +void loop() { + // Handle LED changes safely outside ISR + if (ledStateChanged) { + ledStateChanged = false; + ledState = !ledState; + digitalWrite(LED_PIN, ledState); + } + + // Report button events + if (buttonPressed) { + // Handle press event + } + if (buttonReleased) { + // Handle release event + } +} +``` + +## Expected Output + +``` +ESP32 Lambda FunctionalInterrupt Example +======================================== +Setting up CHANGE mode lambda for LED toggle + +Lambda interrupt configured on Pin 0 (CHANGE mode) +Debounce delay: 50 ms + +Press the button to toggle the LED! +Button press (FALLING edge) will toggle the LED. +Button release (RISING edge) will be detected and reported. +Button includes debouncing to prevent mechanical bounce issues. + +==> Button PRESSED! Count: 1, LED: ON (FALLING edge) +==> Button RELEASED! Count: 1 (RISING edge) +==> Button PRESSED! Count: 2, LED: OFF (FALLING edge) +==> Button RELEASED! Count: 2 (RISING edge) +``` + +## Pin Configuration + +The example uses these default pins: + +- `BUTTON_PIN`: BOOT_PIN (automatically assigned by the Arduino Core) +- `LED_PIN`: LED_BUILTIN (may not be available for your board - please verify it) From ce7ef9c2ba61deff89d86913a966d33c070fb3a4 Mon Sep 17 00:00:00 2001 From: Me No Dev Date: Wed, 16 Jul 2025 13:42:07 +0300 Subject: [PATCH 111/173] IDF release/v5.5 cf8dad07 (#11601) IDF release/v5.5 cf8dad07 --- package/package_esp32_index.template.json | 68 +++++++++++------------ 1 file changed, 34 insertions(+), 34 deletions(-) diff --git a/package/package_esp32_index.template.json b/package/package_esp32_index.template.json index 9c2c754504e..6e2dd0ab337 100644 --- a/package/package_esp32_index.template.json +++ b/package/package_esp32_index.template.json @@ -51,7 +51,7 @@ { "packager": "esp32", "name": "esp32-arduino-libs", - "version": "idf-release_v5.5-adb3f2a5-v1" + "version": "idf-release_v5.5-cf8dad07-v1" }, { "packager": "esp32", @@ -104,63 +104,63 @@ "tools": [ { "name": "esp32-arduino-libs", - "version": "idf-release_v5.5-adb3f2a5-v1", + "version": "idf-release_v5.5-cf8dad07-v1", "systems": [ { "host": "i686-mingw32", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-adb3f2a5-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-adb3f2a5-v1.zip", - "checksum": "SHA-256:c7bdda06e7ddae51880fc0e1c76114a8e766a9d682e3645e7794f7126b895a94", - "size": "430508851" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-cf8dad07-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-cf8dad07-v1.zip", + "checksum": "SHA-256:c8310c661871a5d82b8b09a2b8d792936bf9a5023ed6dcb683ff93ff9ea4aaa7", + "size": "430423461" }, { "host": "x86_64-mingw32", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-adb3f2a5-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-adb3f2a5-v1.zip", - "checksum": "SHA-256:c7bdda06e7ddae51880fc0e1c76114a8e766a9d682e3645e7794f7126b895a94", - "size": "430508851" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-cf8dad07-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-cf8dad07-v1.zip", + "checksum": "SHA-256:c8310c661871a5d82b8b09a2b8d792936bf9a5023ed6dcb683ff93ff9ea4aaa7", + "size": "430423461" }, { "host": "arm64-apple-darwin", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-adb3f2a5-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-adb3f2a5-v1.zip", - "checksum": "SHA-256:c7bdda06e7ddae51880fc0e1c76114a8e766a9d682e3645e7794f7126b895a94", - "size": "430508851" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-cf8dad07-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-cf8dad07-v1.zip", + "checksum": "SHA-256:c8310c661871a5d82b8b09a2b8d792936bf9a5023ed6dcb683ff93ff9ea4aaa7", + "size": "430423461" }, { "host": "x86_64-apple-darwin", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-adb3f2a5-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-adb3f2a5-v1.zip", - "checksum": "SHA-256:c7bdda06e7ddae51880fc0e1c76114a8e766a9d682e3645e7794f7126b895a94", - "size": "430508851" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-cf8dad07-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-cf8dad07-v1.zip", + "checksum": "SHA-256:c8310c661871a5d82b8b09a2b8d792936bf9a5023ed6dcb683ff93ff9ea4aaa7", + "size": "430423461" }, { "host": "x86_64-pc-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-adb3f2a5-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-adb3f2a5-v1.zip", - "checksum": "SHA-256:c7bdda06e7ddae51880fc0e1c76114a8e766a9d682e3645e7794f7126b895a94", - "size": "430508851" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-cf8dad07-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-cf8dad07-v1.zip", + "checksum": "SHA-256:c8310c661871a5d82b8b09a2b8d792936bf9a5023ed6dcb683ff93ff9ea4aaa7", + "size": "430423461" }, { "host": "i686-pc-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-adb3f2a5-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-adb3f2a5-v1.zip", - "checksum": "SHA-256:c7bdda06e7ddae51880fc0e1c76114a8e766a9d682e3645e7794f7126b895a94", - "size": "430508851" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-cf8dad07-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-cf8dad07-v1.zip", + "checksum": "SHA-256:c8310c661871a5d82b8b09a2b8d792936bf9a5023ed6dcb683ff93ff9ea4aaa7", + "size": "430423461" }, { "host": "aarch64-linux-gnu", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-adb3f2a5-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-adb3f2a5-v1.zip", - "checksum": "SHA-256:c7bdda06e7ddae51880fc0e1c76114a8e766a9d682e3645e7794f7126b895a94", - "size": "430508851" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-cf8dad07-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-cf8dad07-v1.zip", + "checksum": "SHA-256:c8310c661871a5d82b8b09a2b8d792936bf9a5023ed6dcb683ff93ff9ea4aaa7", + "size": "430423461" }, { "host": "arm-linux-gnueabihf", - "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-adb3f2a5-v1.zip", - "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-adb3f2a5-v1.zip", - "checksum": "SHA-256:c7bdda06e7ddae51880fc0e1c76114a8e766a9d682e3645e7794f7126b895a94", - "size": "430508851" + "url": "https://github.com/espressif/esp32-arduino-lib-builder/releases/download/idf-release_v5.5/esp32-arduino-libs-idf-release_v5.5-cf8dad07-v1.zip", + "archiveFileName": "esp32-arduino-libs-idf-release_v5.5-cf8dad07-v1.zip", + "checksum": "SHA-256:c8310c661871a5d82b8b09a2b8d792936bf9a5023ed6dcb683ff93ff9ea4aaa7", + "size": "430423461" } ] }, From ac05f18720e83d1e9deee24db7ab13718219b4d6 Mon Sep 17 00:00:00 2001 From: "pre-commit-ci-lite[bot]" <117423508+pre-commit-ci-lite[bot]@users.noreply.github.com> Date: Wed, 16 Jul 2025 12:12:43 +0000 Subject: [PATCH 112/173] ci(pre-commit): Apply automatic fixes --- cores/esp32/esp32-hal-cpu.c | 3 ++- libraries/SPI/src/SPI.cpp | 3 ++- 2 files changed, 4 insertions(+), 2 deletions(-) diff --git a/cores/esp32/esp32-hal-cpu.c b/cores/esp32/esp32-hal-cpu.c index d6bc6b0f191..e9113da4219 100644 --- a/cores/esp32/esp32-hal-cpu.c +++ b/cores/esp32/esp32-hal-cpu.c @@ -19,7 +19,8 @@ #include "esp_attr.h" #include "esp_log.h" #include "soc/rtc.h" -#if !defined(CONFIG_IDF_TARGET_ESP32C2) && !defined(CONFIG_IDF_TARGET_ESP32C6) && !defined(CONFIG_IDF_TARGET_ESP32H2) && !defined(CONFIG_IDF_TARGET_ESP32P4) && !defined(CONFIG_IDF_TARGET_ESP32C5) +#if !defined(CONFIG_IDF_TARGET_ESP32C2) && !defined(CONFIG_IDF_TARGET_ESP32C6) && !defined(CONFIG_IDF_TARGET_ESP32H2) && !defined(CONFIG_IDF_TARGET_ESP32P4) \ + && !defined(CONFIG_IDF_TARGET_ESP32C5) #include "soc/rtc_cntl_reg.h" #include "soc/syscon_reg.h" #endif diff --git a/libraries/SPI/src/SPI.cpp b/libraries/SPI/src/SPI.cpp index 0492065b798..6229f887553 100644 --- a/libraries/SPI/src/SPI.cpp +++ b/libraries/SPI/src/SPI.cpp @@ -84,7 +84,8 @@ bool SPIClass::begin(int8_t sck, int8_t miso, int8_t mosi, int8_t ss) { _miso = (_spi_num == FSPI) ? MISO : -1; _mosi = (_spi_num == FSPI) ? MOSI : -1; _ss = (_spi_num == FSPI) ? SS : -1; -#elif CONFIG_IDF_TARGET_ESP32C2 || CONFIG_IDF_TARGET_ESP32C3 || CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 || CONFIG_IDF_TARGET_ESP32C5 +#elif CONFIG_IDF_TARGET_ESP32C2 || CONFIG_IDF_TARGET_ESP32C3 || CONFIG_IDF_TARGET_ESP32C6 || CONFIG_IDF_TARGET_ESP32H2 || CONFIG_IDF_TARGET_ESP32P4 \ + || CONFIG_IDF_TARGET_ESP32C5 _sck = SCK; _miso = MISO; _mosi = MOSI; From 6015fd73e0d9d065c80fc79ef0f35d3273a217de Mon Sep 17 00:00:00 2001 From: Sugar Glider Date: Wed, 16 Jul 2025 13:53:36 -0300 Subject: [PATCH 113/173] feat(openthread): native API extension (#11598) * feat(openthread): native API extension * fix(openthread): wrong return type and parameter * fix(openthread): wrong field reference * fix(openthread): CR/LF fix * feat(openthread): print leader RLOC information * feat(openthread): code improvements * ci(pre-commit): Apply automatic fixes --------- Co-authored-by: pre-commit-ci-lite[bot] <117423508+pre-commit-ci-lite[bot]@users.noreply.github.com> --- .../LeaderNode/LeaderNode.ino | 85 +++- .../RouterNode/RouterNode.ino | 61 ++- libraries/OpenThread/keywords.txt | 18 + libraries/OpenThread/src/OThread.cpp | 363 +++++++++++++++++- libraries/OpenThread/src/OThread.h | 63 ++- 5 files changed, 565 insertions(+), 25 deletions(-) diff --git a/libraries/OpenThread/examples/Native/SimpleThreadNetwork/LeaderNode/LeaderNode.ino b/libraries/OpenThread/examples/Native/SimpleThreadNetwork/LeaderNode/LeaderNode.ino index dfea9776838..b3c4091e1dc 100644 --- a/libraries/OpenThread/examples/Native/SimpleThreadNetwork/LeaderNode/LeaderNode.ino +++ b/libraries/OpenThread/examples/Native/SimpleThreadNetwork/LeaderNode/LeaderNode.ino @@ -3,6 +3,9 @@ OpenThread threadLeaderNode; DataSet dataset; +// Track last known device role for state change detection +ot_device_role_t lastKnownRole = OT_ROLE_DISABLED; + void setup() { Serial.begin(115200); @@ -27,8 +30,84 @@ void setup() { } void loop() { - // Print network information every 5 seconds - Serial.println("=============================================="); - threadLeaderNode.otPrintNetworkInformation(Serial); + // Get current device role + ot_device_role_t currentRole = threadLeaderNode.otGetDeviceRole(); + + // Only print network information when not detached + if (currentRole != OT_ROLE_DETACHED && currentRole != OT_ROLE_DISABLED) { + Serial.println("=============================================="); + Serial.println("OpenThread Network Information:"); + + // Basic network information + Serial.printf("Role: %s\r\n", threadLeaderNode.otGetStringDeviceRole()); + Serial.printf("RLOC16: 0x%04x\r\n", threadLeaderNode.getRloc16()); + Serial.printf("Network Name: %s\r\n", threadLeaderNode.getNetworkName().c_str()); + Serial.printf("Channel: %d\r\n", threadLeaderNode.getChannel()); + Serial.printf("PAN ID: 0x%04x\r\n", threadLeaderNode.getPanId()); + + // Extended PAN ID + const uint8_t *extPanId = threadLeaderNode.getExtendedPanId(); + if (extPanId) { + Serial.print("Extended PAN ID: "); + for (int i = 0; i < OT_EXT_PAN_ID_SIZE; i++) { + Serial.printf("%02x", extPanId[i]); + } + Serial.println(); + } + + // Network Key + const uint8_t *networkKey = threadLeaderNode.getNetworkKey(); + if (networkKey) { + Serial.print("Network Key: "); + for (int i = 0; i < OT_NETWORK_KEY_SIZE; i++) { + Serial.printf("%02x", networkKey[i]); + } + Serial.println(); + } + + // Mesh Local EID + IPAddress meshLocalEid = threadLeaderNode.getMeshLocalEid(); + Serial.printf("Mesh Local EID: %s\r\n", meshLocalEid.toString().c_str()); + + // Leader RLOC + IPAddress leaderRloc = threadLeaderNode.getLeaderRloc(); + Serial.printf("Leader RLOC: %s\r\n", leaderRloc.toString().c_str()); + + // Node RLOC + IPAddress nodeRloc = threadLeaderNode.getRloc(); + Serial.printf("Node RLOC: %s\r\n", nodeRloc.toString().c_str()); + + // Demonstrate address listing with two different methods: + // Method 1: Unicast addresses using counting API (individual access) + Serial.println("\r\n--- Unicast Addresses (Using Count + Index API) ---"); + size_t unicastCount = threadLeaderNode.getUnicastAddressCount(); + for (size_t i = 0; i < unicastCount; i++) { + IPAddress addr = threadLeaderNode.getUnicastAddress(i); + Serial.printf(" [%zu]: %s\r\n", i, addr.toString().c_str()); + } + + // Method 2: Multicast addresses using std::vector (bulk access) + Serial.println("\r\n--- Multicast Addresses (Using std::vector API) ---"); + std::vector allMulticast = threadLeaderNode.getAllMulticastAddresses(); + for (size_t i = 0; i < allMulticast.size(); i++) { + Serial.printf(" [%zu]: %s\r\n", i, allMulticast[i].toString().c_str()); + } + + // Check for role change and clear cache if needed (only when active) + if (currentRole != lastKnownRole) { + Serial.printf( + "Role changed from %s to %s - clearing address cache\r\n", (lastKnownRole < 5) ? otRoleString[lastKnownRole] : "Unknown", + threadLeaderNode.otGetStringDeviceRole() + ); + threadLeaderNode.clearAllAddressCache(); + lastKnownRole = currentRole; + } + } else { + Serial.printf("Thread Node Status: %s - Waiting for thread network start...\r\n", threadLeaderNode.otGetStringDeviceRole()); + + // Update role tracking even when detached/disabled, but don't clear cache + lastKnownRole = currentRole; + } + delay(5000); } diff --git a/libraries/OpenThread/examples/Native/SimpleThreadNetwork/RouterNode/RouterNode.ino b/libraries/OpenThread/examples/Native/SimpleThreadNetwork/RouterNode/RouterNode.ino index 5ffa535ad51..a8959792f5b 100644 --- a/libraries/OpenThread/examples/Native/SimpleThreadNetwork/RouterNode/RouterNode.ino +++ b/libraries/OpenThread/examples/Native/SimpleThreadNetwork/RouterNode/RouterNode.ino @@ -22,8 +22,63 @@ void setup() { } void loop() { - // Print network information every 5 seconds - Serial.println("=============================================="); - threadChildNode.otPrintNetworkInformation(Serial); + // Get current device role + ot_device_role_t currentRole = threadChildNode.otGetDeviceRole(); + + // Only print detailed network information when node is active + if (currentRole != OT_ROLE_DETACHED && currentRole != OT_ROLE_DISABLED) { + Serial.println("=============================================="); + Serial.println("OpenThread Network Information (Active Dataset):"); + + // Get and display the current active dataset + const DataSet &activeDataset = threadChildNode.getCurrentDataSet(); + + Serial.printf("Role: %s\r\n", threadChildNode.otGetStringDeviceRole()); + Serial.printf("RLOC16: 0x%04x\r\n", threadChildNode.getRloc16()); + + // Dataset information + Serial.printf("Network Name: %s\r\n", activeDataset.getNetworkName()); + Serial.printf("Channel: %d\r\n", activeDataset.getChannel()); + Serial.printf("PAN ID: 0x%04x\r\n", activeDataset.getPanId()); + + // Extended PAN ID from dataset + const uint8_t *extPanId = activeDataset.getExtendedPanId(); + if (extPanId) { + Serial.print("Extended PAN ID: "); + for (int i = 0; i < OT_EXT_PAN_ID_SIZE; i++) { + Serial.printf("%02x", extPanId[i]); + } + Serial.println(); + } + + // Network Key from dataset + const uint8_t *networkKey = activeDataset.getNetworkKey(); + if (networkKey) { + Serial.print("Network Key: "); + for (int i = 0; i < OT_NETWORK_KEY_SIZE; i++) { + Serial.printf("%02x", networkKey[i]); + } + Serial.println(); + } + + // Additional runtime information + IPAddress meshLocalEid = threadChildNode.getMeshLocalEid(); + Serial.printf("Mesh Local EID: %s\r\n", meshLocalEid.toString().c_str()); + + IPAddress nodeRloc = threadChildNode.getRloc(); + Serial.printf("Node RLOC: %s\r\n", nodeRloc.toString().c_str()); + + IPAddress leaderRloc = threadChildNode.getLeaderRloc(); + Serial.printf("Leader RLOC: %s\r\n", leaderRloc.toString().c_str()); + + Serial.println(); + + } else { + Serial.println("=============================================="); + Serial.printf("Thread Node Status: %s - Waiting for thread network start...\r\n", threadChildNode.otGetStringDeviceRole()); + + Serial.println(); + } + delay(5000); } diff --git a/libraries/OpenThread/keywords.txt b/libraries/OpenThread/keywords.txt index b62c2c23ddc..99821ce401c 100644 --- a/libraries/OpenThread/keywords.txt +++ b/libraries/OpenThread/keywords.txt @@ -13,6 +13,7 @@ OpenThread KEYWORD1 DataSet KEYWORD1 ot_cmd_return_t KEYWORD1 ot_device_role_t KEYWORD1 +OnReceiveCb_t KEYWORD1 ####################################### # Methods and Functions (KEYWORD2) @@ -59,6 +60,23 @@ stop KEYWORD2 networkInterfaceUp KEYWORD2 networkInterfaceDown KEYWORD2 commitDataSet KEYWORD2 +getInstance KEYWORD2 +getCurrentDataSet KEYWORD2 +getMeshLocalPrefix KEYWORD2 +getMeshLocalEid KEYWORD2 +getLeaderRloc KEYWORD2 +getRloc KEYWORD2 +getRloc16 KEYWORD2 +getUnicastAddressCount KEYWORD2 +getUnicastAddress KEYWORD2 +getAllUnicastAddresses KEYWORD2 +getMulticastAddressCount KEYWORD2 +getMulticastAddress KEYWORD2 +getAllMulticastAddresses KEYWORD2 +clearUnicastAddressCache KEYWORD2 +clearMulticastAddressCache KEYWORD2 +clearAllAddressCache KEYWORD2 +end KEYWORD2 ####################################### # Constants (LITERAL1) diff --git a/libraries/OpenThread/src/OThread.cpp b/libraries/OpenThread/src/OThread.cpp index 7714d870cce..87b748d104d 100644 --- a/libraries/OpenThread/src/OThread.cpp +++ b/libraries/OpenThread/src/OThread.cpp @@ -2,6 +2,8 @@ #if SOC_IEEE802154_SUPPORTED #if CONFIG_OPENTHREAD_ENABLED +#include "IPAddress.h" +#include #include "esp_err.h" #include "esp_event.h" #include "esp_netif.h" @@ -132,16 +134,29 @@ const otOperationalDataset &DataSet::getDataset() const { } void DataSet::setNetworkName(const char *name) { - strncpy(mDataset.mNetworkName.m8, name, sizeof(mDataset.mNetworkName.m8)); + if (!name) { + log_w("Network name is null"); + return; + } + // char m8[OT_NETWORK_KEY_SIZE + 1] bytes space by definition + strncpy(mDataset.mNetworkName.m8, name, OT_NETWORK_KEY_SIZE); mDataset.mComponents.mIsNetworkNamePresent = true; } void DataSet::setExtendedPanId(const uint8_t *extPanId) { + if (!extPanId) { + log_w("Extended PAN ID is null"); + return; + } memcpy(mDataset.mExtendedPanId.m8, extPanId, OT_EXT_PAN_ID_SIZE); mDataset.mComponents.mIsExtendedPanIdPresent = true; } void DataSet::setNetworkKey(const uint8_t *key) { + if (!key) { + log_w("Network key is null"); + return; + } memcpy(mDataset.mNetworkKey.m8, key, OT_NETWORK_KEY_SIZE); mDataset.mComponents.mIsNetworkKeyPresent = true; } @@ -181,10 +196,18 @@ void DataSet::apply(otInstance *instance) { } // OpenThread Implementation -bool OpenThread::otStarted = false; +bool OpenThread::otStarted; +otInstance *OpenThread::mInstance; +DataSet OpenThread::mCurrentDataset; +otNetworkKey OpenThread::mNetworkKey; -otInstance *OpenThread::mInstance = nullptr; -OpenThread::OpenThread() {} +OpenThread::OpenThread() { + // static initialization (node data and stack starting information) + otStarted = false; + mCurrentDataset.clear(); // Initialize the current dataset + memset(&mNetworkKey, 0, sizeof(mNetworkKey)); // Initialize the network key + mInstance = nullptr; +} OpenThread::~OpenThread() { end(); @@ -214,13 +237,7 @@ void OpenThread::begin(bool OThreadAutoStart) { return; } log_d("OpenThread task created successfully"); - // get the OpenThread instance that will be used for all operations - mInstance = esp_openthread_get_instance(); - if (!mInstance) { - log_e("Error: Failed to initialize OpenThread instance"); - end(); - return; - } + // starts Thread with default dataset from NVS or from IDF default settings if (OThreadAutoStart) { otOperationalDatasetTlvs dataset; @@ -238,23 +255,46 @@ void OpenThread::begin(bool OThreadAutoStart) { log_i("AUTO start OpenThread done"); } } + + // get the OpenThread instance that will be used for all operations + mInstance = esp_openthread_get_instance(); + if (!mInstance) { + log_e("Error: Failed to initialize OpenThread instance"); + end(); + return; + } + otStarted = true; } void OpenThread::end() { + if (!otStarted) { + log_w("OpenThread already stopped"); + return; + } + if (s_ot_task != NULL) { vTaskDelete(s_ot_task); s_ot_task = NULL; - // Clean up - esp_openthread_deinit(); - esp_openthread_netif_glue_deinit(); -#if CONFIG_LWIP_HOOK_IP6_INPUT_CUSTOM - ot_lwip_netif = NULL; -#endif + } + + // Clean up in reverse order of initialization + if (openthread_netif != NULL) { esp_netif_destroy(openthread_netif); - esp_vfs_eventfd_unregister(); + openthread_netif = NULL; } + + esp_openthread_netif_glue_deinit(); + esp_openthread_deinit(); + esp_vfs_eventfd_unregister(); + +#if CONFIG_LWIP_HOOK_IP6_INPUT_CUSTOM + ot_lwip_netif = NULL; +#endif + + mInstance = nullptr; otStarted = false; + log_d("OpenThread ended successfully"); } void OpenThread::start() { @@ -262,6 +302,7 @@ void OpenThread::start() { log_w("Error: OpenThread instance not initialized"); return; } + clearAllAddressCache(); // Clear cache when starting network otThreadSetEnabled(mInstance, true); log_d("Thread network started"); } @@ -271,6 +312,7 @@ void OpenThread::stop() { log_w("Error: OpenThread instance not initialized"); return; } + clearAllAddressCache(); // Clear cache when stopping network otThreadSetEnabled(mInstance, false); log_d("Thread network stopped"); } @@ -285,6 +327,7 @@ void OpenThread::networkInterfaceUp() { if (error != OT_ERROR_NONE) { log_e("Error: Failed to enable Thread interface (error code: %d)\n", error); } + clearAllAddressCache(); // Clear cache when interface comes up log_d("OpenThread Network Interface is up"); } @@ -312,6 +355,7 @@ void OpenThread::commitDataSet(const DataSet &dataset) { log_e("Error: Failed to commit dataset (error code: %d)\n", error); return; } + clearAllAddressCache(); // Clear cache when dataset changes log_d("Dataset committed successfully"); } @@ -360,6 +404,289 @@ void OpenThread::otPrintNetworkInformation(Stream &output) { output.println(); } +// Get the Node Network Name +String OpenThread::getNetworkName() const { + if (!mInstance) { + log_w("Error: OpenThread instance not initialized"); + return String(); // Return empty String, not nullptr + } + const char *networkName = otThreadGetNetworkName(mInstance); + return networkName ? String(networkName) : String(); +} + +// Get the Node Extended PAN ID +const uint8_t *OpenThread::getExtendedPanId() const { + if (!mInstance) { + log_w("Error: OpenThread instance not initialized"); + return nullptr; + } + const otExtendedPanId *extPanId = otThreadGetExtendedPanId(mInstance); + return extPanId ? extPanId->m8 : nullptr; +} + +// Get the Node Network Key +const uint8_t *OpenThread::getNetworkKey() const { + if (!mInstance) { + log_w("Error: OpenThread instance not initialized"); + return nullptr; + } + otThreadGetNetworkKey(mInstance, &mNetworkKey); + return mNetworkKey.m8; +} + +// Get the Node Channel +uint8_t OpenThread::getChannel() const { + if (!mInstance) { + log_w("Error: OpenThread instance not initialized"); + return 0; + } + return otLinkGetChannel(mInstance); +} + +// Get the Node PAN ID +uint16_t OpenThread::getPanId() const { + if (!mInstance) { + log_w("Error: OpenThread instance not initialized"); + return 0; + } + return otLinkGetPanId(mInstance); +} + +// Get the OpenThread instance +otInstance *OpenThread::getInstance() { + if (!mInstance) { + log_w("Error: OpenThread instance not initialized"); + return nullptr; + } + return mInstance; +} + +// Get the current dataset +const DataSet &OpenThread::getCurrentDataSet() const { + + if (!mInstance) { + log_w("Error: OpenThread instance not initialized"); + mCurrentDataset.clear(); + return mCurrentDataset; + } + + otOperationalDataset dataset; + otError error = otDatasetGetActive(mInstance, &dataset); + + if (error == OT_ERROR_NONE) { + mCurrentDataset.clear(); + + if (dataset.mComponents.mIsNetworkNamePresent) { + mCurrentDataset.setNetworkName(dataset.mNetworkName.m8); + } + if (dataset.mComponents.mIsExtendedPanIdPresent) { + mCurrentDataset.setExtendedPanId(dataset.mExtendedPanId.m8); + } + if (dataset.mComponents.mIsNetworkKeyPresent) { + mCurrentDataset.setNetworkKey(dataset.mNetworkKey.m8); + } + if (dataset.mComponents.mIsChannelPresent) { + mCurrentDataset.setChannel(dataset.mChannel); + } + if (dataset.mComponents.mIsPanIdPresent) { + mCurrentDataset.setPanId(dataset.mPanId); + } + } else { + log_w("Failed to get active dataset (error: %d)", error); + mCurrentDataset.clear(); + } + + return mCurrentDataset; +} + +// Get the Mesh Local Prefix +const otMeshLocalPrefix *OpenThread::getMeshLocalPrefix() const { + if (!mInstance) { + log_w("Error: OpenThread instance not initialized"); + return nullptr; + } + return otThreadGetMeshLocalPrefix(mInstance); +} + +// Get the Mesh-Local EID +IPAddress OpenThread::getMeshLocalEid() const { + if (!mInstance) { + log_w("Error: OpenThread instance not initialized"); + return IPAddress(IPv6); // Return empty IPv6 address + } + const otIp6Address *otAddr = otThreadGetMeshLocalEid(mInstance); + if (!otAddr) { + log_w("Failed to get Mesh Local EID"); + return IPAddress(IPv6); + } + return IPAddress(IPv6, otAddr->mFields.m8); +} + +// Get the Thread Leader RLOC +IPAddress OpenThread::getLeaderRloc() const { + if (!mInstance) { + log_w("Error: OpenThread instance not initialized"); + return IPAddress(IPv6); // Return empty IPv6 address + } + otIp6Address otAddr; + otError error = otThreadGetLeaderRloc(mInstance, &otAddr); + if (error != OT_ERROR_NONE) { + log_w("Failed to get Leader RLOC"); + return IPAddress(IPv6); + } + return IPAddress(IPv6, otAddr.mFields.m8); +} + +// Get the Node RLOC +IPAddress OpenThread::getRloc() const { + if (!mInstance) { + log_w("Error: OpenThread instance not initialized"); + return IPAddress(IPv6); // Return empty IPv6 address + } + const otIp6Address *otAddr = otThreadGetRloc(mInstance); + if (!otAddr) { + log_w("Failed to get Node RLOC"); + return IPAddress(IPv6); + } + return IPAddress(IPv6, otAddr->mFields.m8); +} + +// Get the RLOC16 ID +uint16_t OpenThread::getRloc16() const { + if (!mInstance) { + log_w("Error: OpenThread instance not initialized"); + return 0; + } + return otThreadGetRloc16(mInstance); +} + +// Populate unicast address cache from OpenThread +void OpenThread::populateUnicastAddressCache() const { + if (!mInstance) { + return; + } + + // Clear existing cache + mCachedUnicastAddresses.clear(); + + // Populate unicast addresses cache + const otNetifAddress *addr = otIp6GetUnicastAddresses(mInstance); + while (addr != nullptr) { + mCachedUnicastAddresses.push_back(IPAddress(IPv6, addr->mAddress.mFields.m8)); + addr = addr->mNext; + } + + log_d("Populated unicast address cache with %zu addresses", mCachedUnicastAddresses.size()); +} + +// Populate multicast address cache from OpenThread +void OpenThread::populateMulticastAddressCache() const { + if (!mInstance) { + return; + } + + // Clear existing cache + mCachedMulticastAddresses.clear(); + + // Populate multicast addresses cache + const otNetifMulticastAddress *mAddr = otIp6GetMulticastAddresses(mInstance); + while (mAddr != nullptr) { + mCachedMulticastAddresses.push_back(IPAddress(IPv6, mAddr->mAddress.mFields.m8)); + mAddr = mAddr->mNext; + } + + log_d("Populated multicast address cache with %zu addresses", mCachedMulticastAddresses.size()); +} + +// Clear unicast address cache +void OpenThread::clearUnicastAddressCache() const { + mCachedUnicastAddresses.clear(); + log_d("Cleared unicast address cache"); +} + +// Clear multicast address cache +void OpenThread::clearMulticastAddressCache() const { + mCachedMulticastAddresses.clear(); + log_d("Cleared multicast address cache"); +} + +// Clear all address caches +void OpenThread::clearAllAddressCache() const { + mCachedUnicastAddresses.clear(); + mCachedMulticastAddresses.clear(); + log_d("Cleared all address caches"); +} + +// Get count of unicast addresses +size_t OpenThread::getUnicastAddressCount() const { + // Populate cache if empty + if (mCachedUnicastAddresses.empty()) { + populateUnicastAddressCache(); + } + + return mCachedUnicastAddresses.size(); +} + +// Get unicast address by index +IPAddress OpenThread::getUnicastAddress(size_t index) const { + // Populate cache if empty + if (mCachedUnicastAddresses.empty()) { + populateUnicastAddressCache(); + } + + if (index >= mCachedUnicastAddresses.size()) { + log_w("Unicast address index %zu out of range (max: %zu)", index, mCachedUnicastAddresses.size()); + return IPAddress(IPv6); + } + + return mCachedUnicastAddresses[index]; +} + +// Get all unicast addresses +std::vector OpenThread::getAllUnicastAddresses() const { + // Populate cache if empty + if (mCachedUnicastAddresses.empty()) { + populateUnicastAddressCache(); + } + + return mCachedUnicastAddresses; // Return copy of cached vector +} + +// Get count of multicast addresses +size_t OpenThread::getMulticastAddressCount() const { + // Populate cache if empty + if (mCachedMulticastAddresses.empty()) { + populateMulticastAddressCache(); + } + + return mCachedMulticastAddresses.size(); +} + +// Get multicast address by index +IPAddress OpenThread::getMulticastAddress(size_t index) const { + // Populate cache if empty + if (mCachedMulticastAddresses.empty()) { + populateMulticastAddressCache(); + } + + if (index >= mCachedMulticastAddresses.size()) { + log_w("Multicast address index %zu out of range (max: %zu)", index, mCachedMulticastAddresses.size()); + return IPAddress(IPv6); + } + + return mCachedMulticastAddresses[index]; +} + +// Get all multicast addresses +std::vector OpenThread::getAllMulticastAddresses() const { + // Populate cache if empty + if (mCachedMulticastAddresses.empty()) { + populateMulticastAddressCache(); + } + + return mCachedMulticastAddresses; // Return copy of cached vector +} + OpenThread OThread; #endif /* CONFIG_OPENTHREAD_ENABLED */ diff --git a/libraries/OpenThread/src/OThread.h b/libraries/OpenThread/src/OThread.h index 359d581bb9d..6e21b854574 100644 --- a/libraries/OpenThread/src/OThread.h +++ b/libraries/OpenThread/src/OThread.h @@ -25,6 +25,8 @@ #include #include #include +#include "IPAddress.h" +#include typedef enum { OT_ROLE_DISABLED = 0, ///< The Thread stack is disabled. @@ -96,9 +98,68 @@ class OpenThread { // Set the dataset void commitDataSet(const DataSet &dataset); + // Get the Node Network Name + String getNetworkName() const; + + // Get the Node Extended PAN ID + const uint8_t *getExtendedPanId() const; + + // Get the Node Network Key + const uint8_t *getNetworkKey() const; + + // Get the Node Channel + uint8_t getChannel() const; + + // Get the Node PAN ID + uint16_t getPanId() const; + + // Get the OpenThread instance + otInstance *getInstance(); + + // Get the current dataset + const DataSet &getCurrentDataSet() const; + + // Get the Mesh Local Prefix + const otMeshLocalPrefix *getMeshLocalPrefix() const; + + // Get the Mesh-Local EID + IPAddress getMeshLocalEid() const; + + // Get the Thread Leader RLOC + IPAddress getLeaderRloc() const; + + // Get the Node RLOC + IPAddress getRloc() const; + + // Get the RLOC16 ID + uint16_t getRloc16() const; + + // Address management with caching + size_t getUnicastAddressCount() const; + IPAddress getUnicastAddress(size_t index) const; + std::vector getAllUnicastAddresses() const; + + size_t getMulticastAddressCount() const; + IPAddress getMulticastAddress(size_t index) const; + std::vector getAllMulticastAddresses() const; + + // Cache management + void clearUnicastAddressCache() const; + void clearMulticastAddressCache() const; + void clearAllAddressCache() const; + private: static otInstance *mInstance; - DataSet mCurrentDataSet; + static DataSet mCurrentDataset; // Current dataset being used by the OpenThread instance. + static otNetworkKey mNetworkKey; // Static storage to persist after function return + + // Address caching for performance (user-controlled) + mutable std::vector mCachedUnicastAddresses; + mutable std::vector mCachedMulticastAddresses; + + // Internal cache management + void populateUnicastAddressCache() const; + void populateMulticastAddressCache() const; }; extern OpenThread OThread; From a69c71f6ad771790ed4716fba063930f804806f6 Mon Sep 17 00:00:00 2001 From: me-no-dev Date: Wed, 16 Jul 2025 19:55:53 +0300 Subject: [PATCH 114/173] feat(core): Update core version to 3.3.0 --- .gitlab/workflows/common.yml | 4 ++-- cores/esp32/esp_arduino_version.h | 4 ++-- docs/conf_common.py | 4 ++-- libraries/ArduinoOTA/library.properties | 2 +- libraries/AsyncUDP/library.properties | 2 +- libraries/BLE/library.properties | 2 +- libraries/BluetoothSerial/library.properties | 2 +- libraries/DNSServer/library.properties | 2 +- libraries/EEPROM/library.properties | 2 +- libraries/ESP32/library.properties | 2 +- libraries/ESP_I2S/library.properties | 2 +- libraries/ESP_NOW/library.properties | 2 +- libraries/ESP_SR/library.properties | 2 +- libraries/ESPmDNS/library.properties | 2 +- libraries/Ethernet/library.properties | 2 +- libraries/FFat/library.properties | 2 +- libraries/FS/library.properties | 2 +- libraries/HTTPClient/library.properties | 2 +- libraries/HTTPUpdate/library.properties | 2 +- libraries/HTTPUpdateServer/library.properties | 2 +- libraries/Insights/library.properties | 2 +- libraries/LittleFS/library.properties | 2 +- libraries/Matter/library.properties | 2 +- libraries/NetBIOS/library.properties | 2 +- libraries/Network/library.properties | 2 +- libraries/NetworkClientSecure/library.properties | 2 +- libraries/OpenThread/library.properties | 2 +- libraries/PPP/library.properties | 2 +- libraries/Preferences/library.properties | 2 +- libraries/RainMaker/library.properties | 2 +- libraries/SD/library.properties | 2 +- libraries/SD_MMC/library.properties | 2 +- libraries/SPI/library.properties | 2 +- libraries/SPIFFS/library.properties | 2 +- libraries/SimpleBLE/library.properties | 2 +- libraries/TFLiteMicro/library.properties | 2 +- libraries/Ticker/library.properties | 2 +- libraries/USB/library.properties | 2 +- libraries/Update/library.properties | 2 +- libraries/WebServer/library.properties | 2 +- libraries/WiFi/library.properties | 2 +- libraries/WiFiProv/library.properties | 2 +- libraries/Wire/library.properties | 2 +- libraries/Zigbee/library.properties | 2 +- package.json | 2 +- platform.txt | 2 +- 46 files changed, 49 insertions(+), 49 deletions(-) diff --git a/.gitlab/workflows/common.yml b/.gitlab/workflows/common.yml index 9086da018ab..611e1d974f4 100644 --- a/.gitlab/workflows/common.yml +++ b/.gitlab/workflows/common.yml @@ -9,8 +9,8 @@ stages: - result variables: - ESP_IDF_VERSION: "5.4" - ESP_ARDUINO_VERSION: "3.2.1" + ESP_IDF_VERSION: "5.5" + ESP_ARDUINO_VERSION: "3.3.0" ############# # `default` # diff --git a/cores/esp32/esp_arduino_version.h b/cores/esp32/esp_arduino_version.h index 97bb3ac794b..120377c61f7 100644 --- a/cores/esp32/esp_arduino_version.h +++ b/cores/esp32/esp_arduino_version.h @@ -21,9 +21,9 @@ extern "C" { /** Major version number (X.x.x) */ #define ESP_ARDUINO_VERSION_MAJOR 3 /** Minor version number (x.X.x) */ -#define ESP_ARDUINO_VERSION_MINOR 2 +#define ESP_ARDUINO_VERSION_MINOR 3 /** Patch version number (x.x.X) */ -#define ESP_ARDUINO_VERSION_PATCH 1 +#define ESP_ARDUINO_VERSION_PATCH 0 /** * Macro to convert ARDUINO version number into an integer diff --git a/docs/conf_common.py b/docs/conf_common.py index af1d615f753..10d4bd715b2 100644 --- a/docs/conf_common.py +++ b/docs/conf_common.py @@ -4,8 +4,8 @@ # Used for substituting variables in the documentation rst_prolog = """ -.. |version| replace:: 3.2.1 -.. |idf_version| replace:: 5.4 +.. |version| replace:: 3.3.0 +.. |idf_version| replace:: 5.5 """ languages = ["en"] diff --git a/libraries/ArduinoOTA/library.properties b/libraries/ArduinoOTA/library.properties index 18de5aa2180..3a3f7e111e0 100644 --- a/libraries/ArduinoOTA/library.properties +++ b/libraries/ArduinoOTA/library.properties @@ -1,5 +1,5 @@ name=ArduinoOTA -version=3.2.1 +version=3.3.0 author=Ivan Grokhotkov and Hristo Gochkov maintainer=Hristo Gochkov sentence=Enables Over The Air upgrades, via wifi and espota.py UDP request/TCP download. diff --git a/libraries/AsyncUDP/library.properties b/libraries/AsyncUDP/library.properties index c64c60d0421..ddf9f79b6d9 100644 --- a/libraries/AsyncUDP/library.properties +++ b/libraries/AsyncUDP/library.properties @@ -1,5 +1,5 @@ name=ESP32 Async UDP -version=3.2.1 +version=3.3.0 author=Me-No-Dev maintainer=Me-No-Dev sentence=Async UDP Library for ESP32 diff --git a/libraries/BLE/library.properties b/libraries/BLE/library.properties index 335ab2fc965..c6fae19a4a1 100644 --- a/libraries/BLE/library.properties +++ b/libraries/BLE/library.properties @@ -1,5 +1,5 @@ name=BLE -version=3.2.1 +version=3.3.0 author=Neil Kolban maintainer=lucasssvaz sentence=BLE functions for ESP32 diff --git a/libraries/BluetoothSerial/library.properties b/libraries/BluetoothSerial/library.properties index 4bc1427e3f8..49211bf3b63 100644 --- a/libraries/BluetoothSerial/library.properties +++ b/libraries/BluetoothSerial/library.properties @@ -1,5 +1,5 @@ name=BluetoothSerial -version=3.2.1 +version=3.3.0 author=Evandro Copercini maintainer=Evandro Copercini sentence=Simple UART to Classical Bluetooth bridge for ESP32 diff --git a/libraries/DNSServer/library.properties b/libraries/DNSServer/library.properties index 42e4c38dc9d..c193b919d02 100644 --- a/libraries/DNSServer/library.properties +++ b/libraries/DNSServer/library.properties @@ -1,5 +1,5 @@ name=DNSServer -version=3.2.1 +version=3.3.0 author=Kristijan Novoselić maintainer=Kristijan Novoselić, sentence=A simple DNS server for ESP32. diff --git a/libraries/EEPROM/library.properties b/libraries/EEPROM/library.properties index ee1caae0792..6d69f52a085 100644 --- a/libraries/EEPROM/library.properties +++ b/libraries/EEPROM/library.properties @@ -1,5 +1,5 @@ name=EEPROM -version=3.2.1 +version=3.3.0 author=Ivan Grokhotkov maintainer=Paolo Becchi sentence=Enables reading and writing data a sequential, addressable FLASH storage diff --git a/libraries/ESP32/library.properties b/libraries/ESP32/library.properties index 0815a69ce6b..e664022388d 100644 --- a/libraries/ESP32/library.properties +++ b/libraries/ESP32/library.properties @@ -1,5 +1,5 @@ name=ESP32 -version=3.2.1 +version=3.3.0 author=Hristo Gochkov, Ivan Grokhtkov maintainer=Hristo Gochkov sentence=ESP32 sketches examples diff --git a/libraries/ESP_I2S/library.properties b/libraries/ESP_I2S/library.properties index ff1ad5d59da..b2763f4e7e8 100644 --- a/libraries/ESP_I2S/library.properties +++ b/libraries/ESP_I2S/library.properties @@ -1,5 +1,5 @@ name=ESP_I2S -version=3.2.1 +version=3.3.0 author=me-no-dev maintainer=me-no-dev sentence=Library for ESP I2S communication diff --git a/libraries/ESP_NOW/library.properties b/libraries/ESP_NOW/library.properties index 426a9464ace..f8e627dbc03 100644 --- a/libraries/ESP_NOW/library.properties +++ b/libraries/ESP_NOW/library.properties @@ -1,5 +1,5 @@ name=ESP_NOW -version=3.2.1 +version=3.3.0 author=me-no-dev maintainer=P-R-O-C-H-Y sentence=Library for ESP_NOW diff --git a/libraries/ESP_SR/library.properties b/libraries/ESP_SR/library.properties index 3b9777bca8f..9d9787b7931 100644 --- a/libraries/ESP_SR/library.properties +++ b/libraries/ESP_SR/library.properties @@ -1,5 +1,5 @@ name=ESP_SR -version=3.2.1 +version=3.3.0 author=me-no-dev maintainer=me-no-dev sentence=Library for ESP Sound Recognition diff --git a/libraries/ESPmDNS/library.properties b/libraries/ESPmDNS/library.properties index b99a5f58e5c..062d3b90b51 100644 --- a/libraries/ESPmDNS/library.properties +++ b/libraries/ESPmDNS/library.properties @@ -1,5 +1,5 @@ name=ESPmDNS -version=3.2.1 +version=3.3.0 author=Hristo Gochkov, Ivan Grokhtkov maintainer=Hristo Gochkov sentence=ESP32 mDNS Library diff --git a/libraries/Ethernet/library.properties b/libraries/Ethernet/library.properties index cd2d8ead018..28f2a8697d9 100644 --- a/libraries/Ethernet/library.properties +++ b/libraries/Ethernet/library.properties @@ -1,5 +1,5 @@ name=Ethernet -version=3.2.1 +version=3.3.0 author=Hristo Gochkov maintainer=Hristo Gochkov sentence=Enables network connection (local and Internet) using the ESP32 Ethernet. diff --git a/libraries/FFat/library.properties b/libraries/FFat/library.properties index 898982536b0..25b8c4e8acd 100644 --- a/libraries/FFat/library.properties +++ b/libraries/FFat/library.properties @@ -1,5 +1,5 @@ name=FFat -version=3.2.1 +version=3.3.0 author=Hristo Gochkov, Ivan Grokhtkov, Larry Bernstone maintainer=Hristo Gochkov sentence=ESP32 FAT on Flash File System diff --git a/libraries/FS/library.properties b/libraries/FS/library.properties index fe86d613bdf..0f05f1134d5 100644 --- a/libraries/FS/library.properties +++ b/libraries/FS/library.properties @@ -1,5 +1,5 @@ name=FS -version=3.2.1 +version=3.3.0 author=Hristo Gochkov, Ivan Grokhtkov maintainer=Hristo Gochkov sentence=ESP32 File System diff --git a/libraries/HTTPClient/library.properties b/libraries/HTTPClient/library.properties index 76da4c857e1..bb5b0936255 100644 --- a/libraries/HTTPClient/library.properties +++ b/libraries/HTTPClient/library.properties @@ -1,5 +1,5 @@ name=HTTPClient -version=3.2.1 +version=3.3.0 author=Markus Sattler maintainer=Markus Sattler sentence=HTTP Client for ESP32 diff --git a/libraries/HTTPUpdate/library.properties b/libraries/HTTPUpdate/library.properties index dfe0474c3f3..88466a3a72e 100644 --- a/libraries/HTTPUpdate/library.properties +++ b/libraries/HTTPUpdate/library.properties @@ -1,5 +1,5 @@ name=HTTPUpdate -version=3.2.1 +version=3.3.0 author=Markus Sattler maintainer=Markus Sattler sentence=Http Update for ESP32 diff --git a/libraries/HTTPUpdateServer/library.properties b/libraries/HTTPUpdateServer/library.properties index 90aa966d27f..c182eeb8d7f 100644 --- a/libraries/HTTPUpdateServer/library.properties +++ b/libraries/HTTPUpdateServer/library.properties @@ -1,5 +1,5 @@ name=HTTPUpdateServer -version=3.2.1 +version=3.3.0 author=Hristo Kapanakov maintainer= sentence=Simple HTTP Update server based on the WebServer diff --git a/libraries/Insights/library.properties b/libraries/Insights/library.properties index c1ad9c72ce1..3ef98d25be6 100644 --- a/libraries/Insights/library.properties +++ b/libraries/Insights/library.properties @@ -1,5 +1,5 @@ name=ESP Insights -version=3.2.1 +version=3.3.0 author=Sanket Wadekar maintainer=Sanket Wadekar sentence=ESP Insights diff --git a/libraries/LittleFS/library.properties b/libraries/LittleFS/library.properties index e00ed49c312..202d8ad4a6d 100644 --- a/libraries/LittleFS/library.properties +++ b/libraries/LittleFS/library.properties @@ -1,5 +1,5 @@ name=LittleFS -version=3.2.1 +version=3.3.0 author= maintainer= sentence=LittleFS for esp32 diff --git a/libraries/Matter/library.properties b/libraries/Matter/library.properties index b601fce0ff5..0b140bfa169 100644 --- a/libraries/Matter/library.properties +++ b/libraries/Matter/library.properties @@ -1,5 +1,5 @@ name=Matter -version=3.2.1 +version=3.3.0 author=Rodrigo Garcia | GitHub @SuGlider maintainer=Rodrigo Garcia sentence=Library for supporting Matter environment on ESP32. diff --git a/libraries/NetBIOS/library.properties b/libraries/NetBIOS/library.properties index a4bbf93c0ed..71d22d6f363 100644 --- a/libraries/NetBIOS/library.properties +++ b/libraries/NetBIOS/library.properties @@ -1,5 +1,5 @@ name=NetBIOS -version=3.2.1 +version=3.3.0 author=Pablo@xpablo.cz maintainer=Hristo Gochkov sentence=Enables NBNS (NetBIOS) name resolution. diff --git a/libraries/Network/library.properties b/libraries/Network/library.properties index f2284981704..f7e04f4de3b 100644 --- a/libraries/Network/library.properties +++ b/libraries/Network/library.properties @@ -1,5 +1,5 @@ name=Networking -version=3.2.1 +version=3.3.0 author=Hristo Gochkov maintainer=Hristo Gochkov sentence=General network management library. diff --git a/libraries/NetworkClientSecure/library.properties b/libraries/NetworkClientSecure/library.properties index 31af7a1bc8c..6ebf4c0ec70 100644 --- a/libraries/NetworkClientSecure/library.properties +++ b/libraries/NetworkClientSecure/library.properties @@ -1,5 +1,5 @@ name=NetworkClientSecure -version=3.2.1 +version=3.3.0 author=Evandro Luis Copercini maintainer=Github Community sentence=Enables secure network connection (local and Internet) using the ESP32 built-in WiFi. diff --git a/libraries/OpenThread/library.properties b/libraries/OpenThread/library.properties index 2687b1dcd1c..550d4eb1627 100644 --- a/libraries/OpenThread/library.properties +++ b/libraries/OpenThread/library.properties @@ -1,5 +1,5 @@ name=OpenThread -version=3.2.1 +version=3.3.0 author=Rodrigo Garcia | GitHub @SuGlider maintainer=Rodrigo Garcia sentence=Library for OpenThread Network on ESP32. diff --git a/libraries/PPP/library.properties b/libraries/PPP/library.properties index c39647cbe56..537708b1261 100644 --- a/libraries/PPP/library.properties +++ b/libraries/PPP/library.properties @@ -1,5 +1,5 @@ name=PPP -version=3.2.1 +version=3.3.0 author=Hristo Gochkov maintainer=Hristo Gochkov sentence=Enables network connection using GSM Modem. diff --git a/libraries/Preferences/library.properties b/libraries/Preferences/library.properties index 6e0d38348c0..0a7e678aa6c 100644 --- a/libraries/Preferences/library.properties +++ b/libraries/Preferences/library.properties @@ -1,5 +1,5 @@ name=Preferences -version=3.2.1 +version=3.3.0 author=Hristo Gochkov maintainer=Hristo Gochkov sentence=Provides friendly access to ESP32's Non-Volatile Storage diff --git a/libraries/RainMaker/library.properties b/libraries/RainMaker/library.properties index 71b4082a0a7..1d72a8faff5 100644 --- a/libraries/RainMaker/library.properties +++ b/libraries/RainMaker/library.properties @@ -1,5 +1,5 @@ name=ESP RainMaker -version=3.2.1 +version=3.3.0 author=Sweety Mhaiske maintainer=Hristo Gochkov sentence=ESP RainMaker Support diff --git a/libraries/SD/library.properties b/libraries/SD/library.properties index cc51196ed54..9d868dce799 100644 --- a/libraries/SD/library.properties +++ b/libraries/SD/library.properties @@ -1,5 +1,5 @@ name=SD -version=3.2.1 +version=3.3.0 author=Arduino, SparkFun maintainer=Arduino sentence=Enables reading and writing on SD cards. For all Arduino boards. diff --git a/libraries/SD_MMC/library.properties b/libraries/SD_MMC/library.properties index 590eb8ebc52..f96ee4377c2 100644 --- a/libraries/SD_MMC/library.properties +++ b/libraries/SD_MMC/library.properties @@ -1,5 +1,5 @@ name=SD_MMC -version=3.2.1 +version=3.3.0 author=Hristo Gochkov, Ivan Grokhtkov maintainer=Hristo Gochkov sentence=ESP32 SDMMC File System diff --git a/libraries/SPI/library.properties b/libraries/SPI/library.properties index 724137030d1..3403d1c5d4f 100644 --- a/libraries/SPI/library.properties +++ b/libraries/SPI/library.properties @@ -1,5 +1,5 @@ name=SPI -version=3.2.1 +version=3.3.0 author=Hristo Gochkov maintainer=Hristo Gochkov sentence=Enables the communication with devices that use the Serial Peripheral Interface (SPI) Bus. For all Arduino boards, BUT Arduino DUE. diff --git a/libraries/SPIFFS/library.properties b/libraries/SPIFFS/library.properties index fc4601e512c..486ec1b4ce6 100644 --- a/libraries/SPIFFS/library.properties +++ b/libraries/SPIFFS/library.properties @@ -1,5 +1,5 @@ name=SPIFFS -version=3.2.1 +version=3.3.0 author=Hristo Gochkov, Ivan Grokhtkov maintainer=Hristo Gochkov sentence=ESP32 SPIFFS File System diff --git a/libraries/SimpleBLE/library.properties b/libraries/SimpleBLE/library.properties index 768449ee1c4..a7f12207afe 100644 --- a/libraries/SimpleBLE/library.properties +++ b/libraries/SimpleBLE/library.properties @@ -1,5 +1,5 @@ name=SimpleBLE -version=3.2.1 +version=3.3.0 author=Hristo Gochkov maintainer=Hristo Gochkov sentence=Provides really simple BLE advertizer with just on and off diff --git a/libraries/TFLiteMicro/library.properties b/libraries/TFLiteMicro/library.properties index 6ad0c32c7d5..d2dc127f5ab 100644 --- a/libraries/TFLiteMicro/library.properties +++ b/libraries/TFLiteMicro/library.properties @@ -1,5 +1,5 @@ name=TFLite Micro -version=3.2.1 +version=3.3.0 author=Sanket Wadekar maintainer=Sanket Wadekar sentence=TensorFlow Lite for Microcontrollers diff --git a/libraries/Ticker/library.properties b/libraries/Ticker/library.properties index 8a2af554906..8795deb22ce 100644 --- a/libraries/Ticker/library.properties +++ b/libraries/Ticker/library.properties @@ -1,5 +1,5 @@ name=Ticker -version=3.2.1 +version=3.3.0 author=Bert Melis maintainer=Hristo Gochkov sentence=Allows to call functions with a given interval. diff --git a/libraries/USB/library.properties b/libraries/USB/library.properties index 677d736a635..4c2c032545e 100644 --- a/libraries/USB/library.properties +++ b/libraries/USB/library.properties @@ -1,5 +1,5 @@ name=USB -version=3.2.1 +version=3.3.0 author=Hristo Gochkov maintainer=Hristo Gochkov sentence=ESP32S2 USB Library diff --git a/libraries/Update/library.properties b/libraries/Update/library.properties index 4c756397aba..5fd633ec358 100644 --- a/libraries/Update/library.properties +++ b/libraries/Update/library.properties @@ -1,5 +1,5 @@ name=Update -version=3.2.1 +version=3.3.0 author=Hristo Gochkov maintainer=Hristo Gochkov sentence=ESP32 Sketch Update Library diff --git a/libraries/WebServer/library.properties b/libraries/WebServer/library.properties index bf6c26c65e7..913dd00e036 100644 --- a/libraries/WebServer/library.properties +++ b/libraries/WebServer/library.properties @@ -1,5 +1,5 @@ name=WebServer -version=3.2.1 +version=3.3.0 author=Ivan Grokhotkov maintainer=Ivan Grokhtkov sentence=Simple web server library diff --git a/libraries/WiFi/library.properties b/libraries/WiFi/library.properties index a282570ff8a..82ccb32b702 100644 --- a/libraries/WiFi/library.properties +++ b/libraries/WiFi/library.properties @@ -1,5 +1,5 @@ name=WiFi -version=3.2.1 +version=3.3.0 author=Hristo Gochkov maintainer=Hristo Gochkov sentence=Enables network connection (local and Internet) using the ESP32 built-in WiFi. diff --git a/libraries/WiFiProv/library.properties b/libraries/WiFiProv/library.properties index 1cc8e4b0f91..1b19186c40b 100644 --- a/libraries/WiFiProv/library.properties +++ b/libraries/WiFiProv/library.properties @@ -1,5 +1,5 @@ name=WiFiProv -version=3.2.1 +version=3.3.0 author=Switi Mhaiske maintainer=Hristo Gochkov sentence=Enables provisioning. diff --git a/libraries/Wire/library.properties b/libraries/Wire/library.properties index 22cc7f26d86..182e98790bc 100644 --- a/libraries/Wire/library.properties +++ b/libraries/Wire/library.properties @@ -1,5 +1,5 @@ name=Wire -version=3.2.1 +version=3.3.0 author=Hristo Gochkov maintainer=Hristo Gochkov sentence=Allows the communication between devices or sensors connected via Two Wire Interface Bus. For esp8266 boards. diff --git a/libraries/Zigbee/library.properties b/libraries/Zigbee/library.properties index d9587f49fd5..dab96b82a61 100644 --- a/libraries/Zigbee/library.properties +++ b/libraries/Zigbee/library.properties @@ -1,5 +1,5 @@ name=Zigbee -version=3.2.1 +version=3.3.0 author=P-R-O-C-H-Y maintainer=Jan Procházka sentence=Enables zigbee connection with the ESP32 diff --git a/package.json b/package.json index 85a15ab3615..4c3e4725a9d 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "framework-arduinoespressif32", - "version": "3.2.1", + "version": "3.3.0", "description": "Arduino Wiring-based Framework for the Espressif ESP32, ESP32-P4, ESP32-S and ESP32-C series of SoCs", "keywords": [ "framework", diff --git a/platform.txt b/platform.txt index 9e98650ca62..82560190c8f 100644 --- a/platform.txt +++ b/platform.txt @@ -1,5 +1,5 @@ name=ESP32 Arduino -version=3.2.1 +version=3.3.0 tools.esp32-arduino-libs.path={runtime.platform.path}/tools/esp32-arduino-libs tools.esp32-arduino-libs.path.windows={runtime.platform.path}\tools\esp32-arduino-libs From 98d309f84a997138e018174c05232e91836e681f Mon Sep 17 00:00:00 2001 From: me-no-dev Date: Fri, 18 Jul 2025 10:44:01 +0300 Subject: [PATCH 115/173] feat(ci): Enable builds on IDF 5.3, 5.4 and 5.5 --- .github/workflows/push.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/push.yml b/.github/workflows/push.yml index 68148fbcff7..f2f2c0b5ca1 100644 --- a/.github/workflows/push.yml +++ b/.github/workflows/push.yml @@ -248,7 +248,7 @@ jobs: # See https://hub.docker.com/r/espressif/idf/tags and # https://docs.espressif.com/projects/esp-idf/en/latest/esp32/api-guides/tools/idf-docker-image.html # for details. - idf_ver: ["release-v5.5"] + idf_ver: ["release-v5.3","release-v5.4","release-v5.5"] idf_target: [ "esp32", From 6cb518448793f0d42b81cf6e8d02662a8846c4fc Mon Sep 17 00:00:00 2001 From: me-no-dev Date: Fri, 18 Jul 2025 11:12:22 +0300 Subject: [PATCH 116/173] fix(build): Fix build for IDF 5.3.3+ and older releases --- cores/esp32/esp32-hal-i2c-slave.c | 8 ++++++-- cores/esp32/esp32-hal-ledc.c | 3 +++ 2 files changed, 9 insertions(+), 2 deletions(-) diff --git a/cores/esp32/esp32-hal-i2c-slave.c b/cores/esp32/esp32-hal-i2c-slave.c index e616934958e..6d69418e787 100644 --- a/cores/esp32/esp32-hal-i2c-slave.c +++ b/cores/esp32/esp32-hal-i2c-slave.c @@ -338,7 +338,9 @@ esp_err_t i2cSlaveInit(uint8_t num, int sda, int scl, uint16_t slaveID, uint32_t } #endif // !defined(CONFIG_IDF_TARGET_ESP32P4) -#if ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 4, 0) +#if (ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 5, 0)) \ + || (ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 4, 2) && ESP_IDF_VERSION < ESP_IDF_VERSION_VAL(5, 5, 0)) \ + || (ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 3, 3) && ESP_IDF_VERSION < ESP_IDF_VERSION_VAL(5, 4, 0)) i2c_ll_set_mode(i2c->dev, I2C_BUS_MODE_SLAVE); i2c_ll_enable_pins_open_drain(i2c->dev, true); i2c_ll_enable_fifo_mode(i2c->dev, true); @@ -366,7 +368,9 @@ esp_err_t i2cSlaveInit(uint8_t num, int sda, int scl, uint16_t slaveID, uint32_t i2c_ll_disable_intr_mask(i2c->dev, I2C_LL_INTR_MASK); i2c_ll_clear_intr_mask(i2c->dev, I2C_LL_INTR_MASK); -#if ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 4, 0) +#if (ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 5, 0)) \ + || (ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 4, 2) && ESP_IDF_VERSION < ESP_IDF_VERSION_VAL(5, 5, 0)) \ + || (ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 3, 3) && ESP_IDF_VERSION < ESP_IDF_VERSION_VAL(5, 4, 0)) i2c_ll_enable_fifo_mode(i2c->dev, true); #else i2c_ll_slave_set_fifo_mode(i2c->dev, true); diff --git a/cores/esp32/esp32-hal-ledc.c b/cores/esp32/esp32-hal-ledc.c index 18b722f80a2..ffb24db4599 100644 --- a/cores/esp32/esp32-hal-ledc.c +++ b/cores/esp32/esp32-hal-ledc.c @@ -88,6 +88,9 @@ static bool find_free_timer(uint8_t speed_mode, uint8_t *timer_num) { } } +#ifndef SOC_LEDC_TIMER_NUM +#define SOC_LEDC_TIMER_NUM 4 +#endif // Find first unused timer for (uint8_t i = 0; i < SOC_LEDC_TIMER_NUM; i++) { if (!(used_timers & (1 << i))) { From 346e7f41386bef558d56a6113d433f824667a278 Mon Sep 17 00:00:00 2001 From: me-no-dev Date: Fri, 18 Jul 2025 11:18:13 +0300 Subject: [PATCH 117/173] fix(build): Enable I2C FIFO mode only on IDF 5.5+ --- cores/esp32/esp32-hal-i2c-slave.c | 4 +--- 1 file changed, 1 insertion(+), 3 deletions(-) diff --git a/cores/esp32/esp32-hal-i2c-slave.c b/cores/esp32/esp32-hal-i2c-slave.c index 6d69418e787..2a357b3a747 100644 --- a/cores/esp32/esp32-hal-i2c-slave.c +++ b/cores/esp32/esp32-hal-i2c-slave.c @@ -368,9 +368,7 @@ esp_err_t i2cSlaveInit(uint8_t num, int sda, int scl, uint16_t slaveID, uint32_t i2c_ll_disable_intr_mask(i2c->dev, I2C_LL_INTR_MASK); i2c_ll_clear_intr_mask(i2c->dev, I2C_LL_INTR_MASK); -#if (ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 5, 0)) \ - || (ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 4, 2) && ESP_IDF_VERSION < ESP_IDF_VERSION_VAL(5, 5, 0)) \ - || (ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 3, 3) && ESP_IDF_VERSION < ESP_IDF_VERSION_VAL(5, 4, 0)) +#if ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 5, 0) i2c_ll_enable_fifo_mode(i2c->dev, true); #else i2c_ll_slave_set_fifo_mode(i2c->dev, true); From 530c1a43fe9425061c396486d744110ce6ca6f83 Mon Sep 17 00:00:00 2001 From: me-no-dev Date: Fri, 18 Jul 2025 11:19:50 +0300 Subject: [PATCH 118/173] fix(build): Enable I2C FIFO mode only on IDF 5.4.2+ --- cores/esp32/esp32-hal-i2c-slave.c | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/cores/esp32/esp32-hal-i2c-slave.c b/cores/esp32/esp32-hal-i2c-slave.c index 2a357b3a747..c8e59653a8f 100644 --- a/cores/esp32/esp32-hal-i2c-slave.c +++ b/cores/esp32/esp32-hal-i2c-slave.c @@ -368,7 +368,7 @@ esp_err_t i2cSlaveInit(uint8_t num, int sda, int scl, uint16_t slaveID, uint32_t i2c_ll_disable_intr_mask(i2c->dev, I2C_LL_INTR_MASK); i2c_ll_clear_intr_mask(i2c->dev, I2C_LL_INTR_MASK); -#if ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 5, 0) +#if ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 4, 2) i2c_ll_enable_fifo_mode(i2c->dev, true); #else i2c_ll_slave_set_fifo_mode(i2c->dev, true); From 000336bad0446d3b8ded72b1d92e5edc407f8f69 Mon Sep 17 00:00:00 2001 From: me-no-dev Date: Fri, 18 Jul 2025 11:25:36 +0300 Subject: [PATCH 119/173] fix(build): Enable I2C FIFO mode only on IDF 5.4.2+ --- cores/esp32/esp32-hal-i2c-slave.c | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/cores/esp32/esp32-hal-i2c-slave.c b/cores/esp32/esp32-hal-i2c-slave.c index c8e59653a8f..79c994869e2 100644 --- a/cores/esp32/esp32-hal-i2c-slave.c +++ b/cores/esp32/esp32-hal-i2c-slave.c @@ -343,7 +343,11 @@ esp_err_t i2cSlaveInit(uint8_t num, int sda, int scl, uint16_t slaveID, uint32_t || (ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 3, 3) && ESP_IDF_VERSION < ESP_IDF_VERSION_VAL(5, 4, 0)) i2c_ll_set_mode(i2c->dev, I2C_BUS_MODE_SLAVE); i2c_ll_enable_pins_open_drain(i2c->dev, true); +#if ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 4, 2) i2c_ll_enable_fifo_mode(i2c->dev, true); +#else + i2c_ll_slave_set_fifo_mode(i2c->dev, true); +#endif #else i2c_ll_slave_init(i2c->dev); i2c_ll_slave_set_fifo_mode(i2c->dev, true); From 211a0ce1438ad7108ded0299238876f22036d946 Mon Sep 17 00:00:00 2001 From: Lucas Saavedra Vaz <32426024+lucasssvaz@users.noreply.github.com> Date: Fri, 18 Jul 2025 09:15:48 -0300 Subject: [PATCH 120/173] fix(ci): Fix artifact names --- .github/workflows/push.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/push.yml b/.github/workflows/push.yml index f2f2c0b5ca1..1523b7231be 100644 --- a/.github/workflows/push.yml +++ b/.github/workflows/push.yml @@ -283,7 +283,7 @@ jobs: uses: actions/upload-artifact@ea165f8d65b6e75b540449e92b4886f43607fa02 # v4.6.2 if: always() with: - name: sdkconfig-${{ matrix.idf_target }} + name: sdkconfig-${{ matrix.idf_ver }}-${{ matrix.idf_target }} path: ./components/arduino-esp32/idf_component_examples/**/sdkconfig # Save artifacts to gh-pages From 3ad17de6aa6f74f7d01bff4197f9f7a7481b2bb4 Mon Sep 17 00:00:00 2001 From: Sugar Glider Date: Fri, 18 Jul 2025 13:02:55 -0300 Subject: [PATCH 121/173] fix(build): make core compatible with IDF 5.3 (#11607) * fix(uart): make it compatible with IDF 5.3 * fix(ci): Fix artifact names * fix(build): Fix ESP_NOW and WiFi for IDF 5.3 * fix(build): Fix NetworkClient for IDF 5.3 * fix(build): Fix WiFi for IDF 5.3 * fix(build): Fix WiFi Captive Portal for IDF 5.3 --------- Co-authored-by: Lucas Saavedra Vaz <32426024+lucasssvaz@users.noreply.github.com> Co-authored-by: me-no-dev --- cores/esp32/esp32-hal-uart.c | 11 +++++++++++ libraries/ESP_NOW/src/ESP32_NOW.cpp | 4 ++++ libraries/Network/src/NetworkClient.cpp | 8 +++----- libraries/WiFi/examples/WiFiScan/WiFiScan.ino | 2 ++ libraries/WiFi/src/AP.cpp | 2 ++ libraries/WiFi/src/WiFiAP.h | 2 ++ libraries/WiFi/src/WiFiGeneric.cpp | 2 ++ libraries/WiFi/src/WiFiGeneric.h | 2 ++ 8 files changed, 28 insertions(+), 5 deletions(-) diff --git a/cores/esp32/esp32-hal-uart.c b/cores/esp32/esp32-hal-uart.c index 6e5a22da9f8..8bd446a8eb8 100644 --- a/cores/esp32/esp32-hal-uart.c +++ b/cores/esp32/esp32-hal-uart.c @@ -389,7 +389,12 @@ static esp_err_t _uartInternalSetPin(uart_port_t uart_num, int tx_io_num, int rx #endif if (tx_rx_same_io || !_uartTrySetIomuxPin(uart_num, rx_io_num, SOC_UART_RX_PIN_IDX)) { if (uart_num < SOC_UART_HP_NUM) { +#if ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 4, 0) gpio_input_enable(rx_io_num); +#else + gpio_func_sel(rx_io_num, PIN_FUNC_GPIO); + gpio_ll_input_enable(&GPIO, rx_io_num); +#endif esp_rom_gpio_connect_in_signal(rx_io_num, UART_PERIPH_SIGNAL(uart_num, SOC_UART_RX_PIN_IDX), 0); } #if SOC_LP_GPIO_MATRIX_SUPPORTED @@ -422,8 +427,14 @@ static esp_err_t _uartInternalSetPin(uart_port_t uart_num, int tx_io_num, int rx if (cts_io_num >= 0 && !_uartTrySetIomuxPin(uart_num, cts_io_num, SOC_UART_CTS_PIN_IDX)) { if (uart_num < SOC_UART_HP_NUM) { +#if ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 4, 0) gpio_pullup_en(cts_io_num); gpio_input_enable(cts_io_num); +#else + gpio_func_sel(cts_io_num, PIN_FUNC_GPIO); + gpio_set_pull_mode(cts_io_num, GPIO_PULLUP_ONLY); + gpio_set_direction(cts_io_num, GPIO_MODE_INPUT); +#endif esp_rom_gpio_connect_in_signal(cts_io_num, UART_PERIPH_SIGNAL(uart_num, SOC_UART_CTS_PIN_IDX), 0); } #if SOC_LP_GPIO_MATRIX_SUPPORTED diff --git a/libraries/ESP_NOW/src/ESP32_NOW.cpp b/libraries/ESP_NOW/src/ESP32_NOW.cpp index d461fe1473d..e61160a16dd 100644 --- a/libraries/ESP_NOW/src/ESP32_NOW.cpp +++ b/libraries/ESP_NOW/src/ESP32_NOW.cpp @@ -9,6 +9,10 @@ #include "esp32-hal.h" #include "esp_wifi.h" +#ifndef ESP_NOW_MAX_DATA_LEN_V2 +#define ESP_NOW_MAX_DATA_LEN_V2 1470 +#endif + static void (*new_cb)(const esp_now_recv_info_t *info, const uint8_t *data, int len, void *arg) = nullptr; static void *new_arg = nullptr; // * tx_arg = nullptr, * rx_arg = nullptr, static bool _esp_now_has_begun = false; diff --git a/libraries/Network/src/NetworkClient.cpp b/libraries/Network/src/NetworkClient.cpp index 89411a42453..355cbc6c361 100644 --- a/libraries/Network/src/NetworkClient.cpp +++ b/libraries/Network/src/NetworkClient.cpp @@ -23,11 +23,9 @@ #include #include -// It is already defined in IDF as: -//#define IN6_IS_ADDR_V4MAPPED(a) ip6_addr_isipv4mappedipv6((ip6_addr_t*)(a)) -//#define ip6_addr_isipv4mappedipv6(ip6addr) (((ip6addr)->addr[0] == 0) && ((ip6addr)->addr[1] == 0) && (((ip6addr)->addr[2]) == PP_HTONL(0x0000FFFFUL))) -// Keeping as a memory of the change. -//#define _IN6_IS_ADDR_V4MAPPED(a) ((((__const uint32_t *)(a))[0] == 0) && (((__const uint32_t *)(a))[1] == 0) && (((__const uint32_t *)(a))[2] == htonl(0xffff))) +#ifndef IN6_IS_ADDR_V4MAPPED +#define IN6_IS_ADDR_V4MAPPED(a) ((((__const uint32_t *)(a))[0] == 0) && (((__const uint32_t *)(a))[1] == 0) && (((__const uint32_t *)(a))[2] == htonl(0xffff))) +#endif #define WIFI_CLIENT_DEF_CONN_TIMEOUT_MS (3000) #define WIFI_CLIENT_MAX_WRITE_RETRY (10) diff --git a/libraries/WiFi/examples/WiFiScan/WiFiScan.ino b/libraries/WiFi/examples/WiFiScan/WiFiScan.ino index 98733adb0bb..d840959dcb2 100644 --- a/libraries/WiFi/examples/WiFiScan/WiFiScan.ino +++ b/libraries/WiFi/examples/WiFiScan/WiFiScan.ino @@ -58,7 +58,9 @@ void loop() { Serial.println("-------------------------------------"); Serial.println("Default wifi band mode scan:"); Serial.println("-------------------------------------"); +#if ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 4, 2) WiFi.setBandMode(WIFI_BAND_MODE_AUTO); +#endif ScanWiFi(); #if CONFIG_SOC_WIFI_SUPPORT_5G // Wait a bit before scanning again. diff --git a/libraries/WiFi/src/AP.cpp b/libraries/WiFi/src/AP.cpp index a649c3898cb..00c3a1e3733 100644 --- a/libraries/WiFi/src/AP.cpp +++ b/libraries/WiFi/src/AP.cpp @@ -305,6 +305,7 @@ bool APClass::enableNAPT(bool enable) { return true; } +#if ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 4, 2) bool APClass::enableDhcpCaptivePortal() { esp_err_t err = ESP_OK; static char captiveportal_uri[32] = { @@ -343,6 +344,7 @@ bool APClass::enableDhcpCaptivePortal() { return true; } +#endif String APClass::SSID(void) const { if (!started()) { diff --git a/libraries/WiFi/src/WiFiAP.h b/libraries/WiFi/src/WiFiAP.h index 2b7ce469801..67cc0f988d3 100644 --- a/libraries/WiFi/src/WiFiAP.h +++ b/libraries/WiFi/src/WiFiAP.h @@ -53,7 +53,9 @@ class APClass : public NetworkInterface { bool bandwidth(wifi_bandwidth_t bandwidth); bool enableNAPT(bool enable = true); +#if ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 4, 2) bool enableDhcpCaptivePortal(); +#endif String SSID(void) const; uint8_t stationCount(); diff --git a/libraries/WiFi/src/WiFiGeneric.cpp b/libraries/WiFi/src/WiFiGeneric.cpp index 171bd293738..66f8908af77 100644 --- a/libraries/WiFi/src/WiFiGeneric.cpp +++ b/libraries/WiFi/src/WiFiGeneric.cpp @@ -781,6 +781,7 @@ wifi_ps_type_t WiFiGenericClass::getSleep() { return _sleepEnabled; } +#if ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 4, 2) /** * control wifi band mode * @param band_mode enum possible band modes @@ -843,6 +844,7 @@ wifi_band_mode_t WiFiGenericClass::getBandMode() { return WIFI_BAND_MODE_2G_ONLY; #endif } +#endif /** * get the current active wifi band diff --git a/libraries/WiFi/src/WiFiGeneric.h b/libraries/WiFi/src/WiFiGeneric.h index ebeecb51e16..cdc1519d30b 100644 --- a/libraries/WiFi/src/WiFiGeneric.h +++ b/libraries/WiFi/src/WiFiGeneric.h @@ -116,8 +116,10 @@ class WiFiGenericClass { bool setTxPower(wifi_power_t power); wifi_power_t getTxPower(); +#if ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 4, 2) bool setBandMode(wifi_band_mode_t band_mode); wifi_band_mode_t getBandMode(); +#endif wifi_band_t getBand(); bool initiateFTM(uint8_t frm_count = 16, uint16_t burst_period = 2, uint8_t channel = 1, const uint8_t *mac = NULL); From 9641de1b12c3ae4980fc1c6b9a2f939a0b6d4934 Mon Sep 17 00:00:00 2001 From: Paula Scharf Date: Thu, 3 Jul 2025 15:22:07 +0200 Subject: [PATCH 122/173] feat(board): add senseBox Eye board --- boards.txt | 190 ++++++++++++++++++ variants/sensebox_eye/bootloader-tinyuf2.bin | Bin 0 -> 22352 bytes .../sensebox_eye/partitions-16MB-tinyuf2.csv | 10 + variants/sensebox_eye/pins_arduino.h | 90 +++++++++ variants/sensebox_eye/tinyuf2.bin | Bin 0 -> 186912 bytes variants/sensebox_eye/variant.cpp | 71 +++++++ 6 files changed, 361 insertions(+) create mode 100644 variants/sensebox_eye/bootloader-tinyuf2.bin create mode 100644 variants/sensebox_eye/partitions-16MB-tinyuf2.csv create mode 100644 variants/sensebox_eye/pins_arduino.h create mode 100644 variants/sensebox_eye/tinyuf2.bin create mode 100644 variants/sensebox_eye/variant.cpp diff --git a/boards.txt b/boards.txt index ac2b1a336e4..2363fe2a63b 100644 --- a/boards.txt +++ b/boards.txt @@ -41309,6 +41309,196 @@ sensebox_mcu_esp32s2.menu.EraseFlash.none.upload.erase_cmd= sensebox_mcu_esp32s2.menu.EraseFlash.all=Enabled sensebox_mcu_esp32s2.menu.EraseFlash.all.upload.erase_cmd=-e +############################################################## +# senseBox Eye + +sensebox_eye.name=senseBox Eye +senseBox_eye.vid.0=0x303A +senseBox_eye.pid.0=0x82D1 +senseBox_eye.vid.1=0x303A +senseBox_eye.pid.1=0x82D2 +senseBox_eye.vid.2=0x303A +senseBox_eye.pid.2=0x82D3 + +sensebox_eye.bootloader.tool=esptool_py +sensebox_eye.bootloader.tool.default=esptool_py + +sensebox_eye.upload.tool=esptool_py +sensebox_eye.upload.tool.default=esptool_py +sensebox_eye.upload.tool.network=esp_ota + +sensebox_eye.upload.maximum_size=1310720 +sensebox_eye.upload.maximum_data_size=327680 +sensebox_eye.upload.flags= +sensebox_eye.upload.extra_flags= +sensebox_eye.upload.use_1200bps_touch=true +sensebox_eye.upload.wait_for_upload_port=true + +sensebox_eye.serial.disableDTR=true +sensebox_eye.serial.disableRTS=true + +sensebox_eye.build.tarch=xtensa +sensebox_eye.build.bootloader_addr=0x0 +sensebox_eye.build.target=esp32s3 +sensebox_eye.build.mcu=esp32s3 +sensebox_eye.build.core=esp32 +sensebox_eye.build.variant=sensebox_eye +sensebox_eye.build.board=sensebox_eye + +sensebox_eye.build.usb_mode=0 +sensebox_eye.build.cdc_on_boot=1 +sensebox_eye.build.msc_on_boot=1 +sensebox_eye.build.dfu_on_boot=0 +sensebox_eye.build.f_cpu=240000000L +sensebox_eye.build.flash_size=16MB (128Mb) +sensebox_eye.build.flash_freq=80m +sensebox_eye.build.flash_mode=dio +sensebox_eye.build.boot=qio +sensebox_eye.build.boot_freq=80m +sensebox_eye.build.partitions=default_16MB +sensebox_eye.build.defines= +sensebox_eye.build.loop_core= +sensebox_eye.build.event_core= +sensebox_eye.build.psram_type=qspi +sensebox_eye.build.memory_type={build.boot}_{build.psram_type} + +sensebox_eye.menu.JTAGAdapter.default=Disabled +sensebox_eye.menu.JTAGAdapter.default.build.copy_jtag_files=0 +sensebox_eye.menu.JTAGAdapter.builtin=Integrated USB JTAG +sensebox_eye.menu.JTAGAdapter.builtin.build.openocdscript=esp32s3-builtin.cfg +sensebox_eye.menu.JTAGAdapter.builtin.build.copy_jtag_files=1 +sensebox_eye.menu.JTAGAdapter.external=FTDI Adapter +sensebox_eye.menu.JTAGAdapter.external.build.openocdscript=esp32s3-ftdi.cfg +sensebox_eye.menu.JTAGAdapter.external.build.copy_jtag_files=1 +sensebox_eye.menu.JTAGAdapter.bridge=ESP USB Bridge +sensebox_eye.menu.JTAGAdapter.bridge.build.openocdscript=esp32s3-bridge.cfg +sensebox_eye.menu.JTAGAdapter.bridge.build.copy_jtag_files=1 + +sensebox_eye.menu.PSRAM.disabled=Disabled +sensebox_eye.menu.PSRAM.disabled.build.defines= +sensebox_eye.menu.PSRAM.disabled.build.psram_type=qspi +sensebox_eye.menu.PSRAM.opi=OPI PSRAM +sensebox_eye.menu.PSRAM.opi.build.defines=-DBOARD_HAS_PSRAM +sensebox_eye.menu.PSRAM.opi.build.psram_type=opi + +sensebox_eye.menu.FlashMode.qio=QIO 80MHz +sensebox_eye.menu.FlashMode.qio.build.flash_mode=dio +sensebox_eye.menu.FlashMode.qio.build.boot=qio +sensebox_eye.menu.FlashMode.qio.build.boot_freq=80m +sensebox_eye.menu.FlashMode.qio.build.flash_freq=80m +sensebox_eye.menu.FlashMode.dio=DIO 80MHz +sensebox_eye.menu.FlashMode.dio.build.flash_mode=dio +sensebox_eye.menu.FlashMode.dio.build.boot=dio +sensebox_eye.menu.FlashMode.dio.build.boot_freq=80m +sensebox_eye.menu.FlashMode.dio.build.flash_freq=80m + +sensebox_eye.menu.FlashSize.16M=16MB (128Mb) +sensebox_eye.menu.FlashSize.16M.build.flash_size=16MB + +sensebox_eye.menu.LoopCore.1=Core 1 +sensebox_eye.menu.LoopCore.1.build.loop_core=-DARDUINO_RUNNING_CORE=1 +sensebox_eye.menu.LoopCore.0=Core 0 +sensebox_eye.menu.LoopCore.0.build.loop_core=-DARDUINO_RUNNING_CORE=0 + +sensebox_eye.menu.EventsCore.1=Core 1 +sensebox_eye.menu.EventsCore.1.build.event_core=-DARDUINO_EVENT_RUNNING_CORE=1 +sensebox_eye.menu.EventsCore.0=Core 0 +sensebox_eye.menu.EventsCore.0.build.event_core=-DARDUINO_EVENT_RUNNING_CORE=0 + +sensebox_eye.menu.USBMode.hwcdc=Hardware CDC and JTAG +sensebox_eye.menu.USBMode.hwcdc.build.usb_mode=1 +sensebox_eye.menu.USBMode.default=USB-OTG (TinyUSB) +sensebox_eye.menu.USBMode.default.build.usb_mode=0 + +sensebox_eye.menu.CDCOnBoot.default=Enabled +sensebox_eye.menu.CDCOnBoot.default.build.cdc_on_boot=1 +sensebox_eye.menu.CDCOnBoot.cdc=Disabled +sensebox_eye.menu.CDCOnBoot.cdc.build.cdc_on_boot=0 + +sensebox_eye.menu.MSCOnBoot.default=Disabled +sensebox_eye.menu.MSCOnBoot.default.build.msc_on_boot=0 +sensebox_eye.menu.MSCOnBoot.msc=Enabled (Requires USB-OTG Mode) +sensebox_eye.menu.MSCOnBoot.msc.build.msc_on_boot=1 + +sensebox_eye.menu.DFUOnBoot.default=Disabled +sensebox_eye.menu.DFUOnBoot.default.build.dfu_on_boot=0 +sensebox_eye.menu.DFUOnBoot.dfu=Enabled (Requires USB-OTG Mode) +sensebox_eye.menu.DFUOnBoot.dfu.build.dfu_on_boot=1 + +sensebox_eye.menu.UploadMode.default=UART0 / Hardware CDC +sensebox_eye.menu.UploadMode.default.upload.use_1200bps_touch=false +sensebox_eye.menu.UploadMode.default.upload.wait_for_upload_port=false +sensebox_eye.menu.UploadMode.cdc=USB-OTG CDC (TinyUSB) +sensebox_eye.menu.UploadMode.cdc.upload.use_1200bps_touch=true +sensebox_eye.menu.UploadMode.cdc.upload.wait_for_upload_port=true + +sensebox_eye.menu.PartitionScheme.default_16MB=Default (6.25MB APP/3.43MB SPIFFS) +sensebox_eye.menu.PartitionScheme.default_16MB.build.partitions=default_16MB +sensebox_eye.menu.PartitionScheme.default_16MB.upload.maximum_size=6553600 +sensebox_eye.menu.PartitionScheme.large_spiffs=Large SPIFFS (4.5MB APP/6.93MB SPIFFS) +sensebox_eye.menu.PartitionScheme.large_spiffs.build.partitions=large_spiffs_16MB +sensebox_eye.menu.PartitionScheme.large_spiffs.upload.maximum_size=4718592 +sensebox_eye.menu.PartitionScheme.app3M_fat9M_16MB=FFAT (3MB APP/9MB FATFS) +sensebox_eye.menu.PartitionScheme.app3M_fat9M_16MB.build.partitions=app3M_fat9M_16MB +sensebox_eye.menu.PartitionScheme.app3M_fat9M_16MB.upload.maximum_size=3145728 +sensebox_eye.menu.PartitionScheme.fatflash=Large FFAT (2MB APP/12.5MB FATFS) +sensebox_eye.menu.PartitionScheme.fatflash.build.partitions=ffat +sensebox_eye.menu.PartitionScheme.fatflash.upload.maximum_size=2097152 +sensebox_eye.menu.PartitionScheme.tinyuf2=TinyUF2 Compatibility (2MB APP/12MB FFAT) +sensebox_eye.menu.PartitionScheme.tinyuf2.build.custom_bootloader=bootloader_tinyuf2 +sensebox_eye.menu.PartitionScheme.tinyuf2.build.custom_partitions=partitions_tinyuf2 +sensebox_eye.menu.PartitionScheme.tinyuf2.upload.maximum_size=2097152 +sensebox_eye.menu.PartitionScheme.tinyuf2.upload.extra_flags=0x410000 "{runtime.platform.path}/variants/{build.variant}/tinyuf2.bin" +sensebox_eye.menu.PartitionScheme.gen4esp32scheme4=Huge App (16MB APP) +sensebox_eye.menu.PartitionScheme.gen4esp32scheme4.build.custom_partitions=gen4esp32_16MBapp +sensebox_eye.menu.PartitionScheme.gen4esp32scheme4.upload.maximum_size=16646144 + +sensebox_eye.menu.CPUFreq.240=240MHz (WiFi) +sensebox_eye.menu.CPUFreq.240.build.f_cpu=240000000L +sensebox_eye.menu.CPUFreq.160=160MHz (WiFi) +sensebox_eye.menu.CPUFreq.160.build.f_cpu=160000000L +sensebox_eye.menu.CPUFreq.80=80MHz (WiFi) +sensebox_eye.menu.CPUFreq.80.build.f_cpu=80000000L +sensebox_eye.menu.CPUFreq.40=40MHz +sensebox_eye.menu.CPUFreq.40.build.f_cpu=40000000L +sensebox_eye.menu.CPUFreq.20=20MHz +sensebox_eye.menu.CPUFreq.20.build.f_cpu=20000000L +sensebox_eye.menu.CPUFreq.10=10MHz +sensebox_eye.menu.CPUFreq.10.build.f_cpu=10000000L + +sensebox_eye.menu.UploadSpeed.921600=921600 +sensebox_eye.menu.UploadSpeed.921600.upload.speed=921600 +sensebox_eye.menu.UploadSpeed.115200=115200 +sensebox_eye.menu.UploadSpeed.115200.upload.speed=115200 +sensebox_eye.menu.UploadSpeed.256000.windows=256000 +sensebox_eye.menu.UploadSpeed.256000.upload.speed=256000 +sensebox_eye.menu.UploadSpeed.230400.windows.upload.speed=256000 +sensebox_eye.menu.UploadSpeed.230400=230400 +sensebox_eye.menu.UploadSpeed.230400.upload.speed=230400 +sensebox_eye.menu.UploadSpeed.460800.linux=460800 +sensebox_eye.menu.UploadSpeed.460800.macosx=460800 +sensebox_eye.menu.UploadSpeed.460800.upload.speed=460800 +sensebox_eye.menu.UploadSpeed.512000.windows=512000 +sensebox_eye.menu.UploadSpeed.512000.upload.speed=512000 + +sensebox_eye.menu.DebugLevel.none=None +sensebox_eye.menu.DebugLevel.none.build.code_debug=0 +sensebox_eye.menu.DebugLevel.error=Error +sensebox_eye.menu.DebugLevel.error.build.code_debug=1 +sensebox_eye.menu.DebugLevel.warn=Warn +sensebox_eye.menu.DebugLevel.warn.build.code_debug=2 +sensebox_eye.menu.DebugLevel.info=Info +sensebox_eye.menu.DebugLevel.info.build.code_debug=3 +sensebox_eye.menu.DebugLevel.debug=Debug +sensebox_eye.menu.DebugLevel.debug.build.code_debug=4 +sensebox_eye.menu.DebugLevel.verbose=Verbose +sensebox_eye.menu.DebugLevel.verbose.build.code_debug=5 + +sensebox_eye.menu.EraseFlash.none=Disabled +sensebox_eye.menu.EraseFlash.none.upload.erase_cmd= +sensebox_eye.menu.EraseFlash.all=Enabled +sensebox_eye.menu.EraseFlash.all.upload.erase_cmd=-e + ############################################################## nano_nora.name=Arduino Nano ESP32 diff --git a/variants/sensebox_eye/bootloader-tinyuf2.bin b/variants/sensebox_eye/bootloader-tinyuf2.bin new file mode 100644 index 0000000000000000000000000000000000000000..c7676c718f2383a78785da38dcffde52ca67eab0 GIT binary patch literal 22352 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 C4c&VH literal 0 HcmV?d00001 diff --git a/variants/sensebox_eye/partitions-16MB-tinyuf2.csv b/variants/sensebox_eye/partitions-16MB-tinyuf2.csv new file mode 100644 index 00000000000..960f3fbbd71 --- /dev/null +++ b/variants/sensebox_eye/partitions-16MB-tinyuf2.csv @@ -0,0 +1,10 @@ +# ESP-IDF Partition Table +# Name, Type, SubType, Offset, Size, Flags +# bootloader.bin,, 0x1000, 32K +# partition table, 0x8000, 4K +nvs, data, nvs, 0x9000, 20K, +otadata, data, ota, 0xe000, 8K, +ota_0, 0, ota_0, 0x10000, 2048K, +ota_1, 0, ota_1, 0x210000, 2048K, +uf2, app, factory,0x410000, 256K, +ffat, data, fat, 0x450000, 11968K, \ No newline at end of file diff --git a/variants/sensebox_eye/pins_arduino.h b/variants/sensebox_eye/pins_arduino.h new file mode 100644 index 00000000000..583cb38283e --- /dev/null +++ b/variants/sensebox_eye/pins_arduino.h @@ -0,0 +1,90 @@ +#ifndef Pins_Arduino_h +#define Pins_Arduino_h + +#include + +#define USB_VID 0x303A +#define USB_PID 0x81B8 +#define USB_MANUFACTURER "senseBox" +#define USB_PRODUCT "Eye ESP32S3" +#define USB_SERIAL "" // Empty string for MAC address + +// Default USB FirmwareMSC Settings +#define USB_FW_MSC_VENDOR_ID "senseBox" // max 8 chars +#define USB_FW_MSC_PRODUCT_ID "Eye ESP32S3" // max 16 chars +#define USB_FW_MSC_PRODUCT_REVISION "1.00" // max 4 chars +#define USB_FW_MSC_VOLUME_NAME "senseBox" // max 11 chars +#define USB_FW_MSC_SERIAL_NUMBER 0x00000000 + +#define PIN_RGB_LED 45 // RGB LED +#define RGBLED_PIN 45 // RGB LED +#define PIN_LED 45 +#define RGBLED_NUM 1 // number of RGB LEDs + +// Default I2C QWIIC-Ports +static const uint8_t SDA = 2; +static const uint8_t SCL = 1; +#define PIN_QWIIC_SDA 2 +#define PIN_QWIIC_SCL 1 + +// IO Pins +#define PIN_IO14 14 +static const uint8_t A14 = PIN_IO14; // Analog +static const uint8_t D14 = PIN_IO14; // Digital +static const uint8_t T14 = PIN_IO14; // Touch +#define PIN_IO48 48 +static const uint8_t A48 = PIN_IO48; // Analog +static const uint8_t D48 = PIN_IO48; // Digital +static const uint8_t T48 = PIN_IO48; // Touch + +// Button +#define PIN_BUTTON 47 + +// UART Port +static const uint8_t TX = 43; +static const uint8_t RX = 44; +#define PIN_UART_TXD 43 +#define PIN_UART_RXD 44 +#define PIN_UART_ENABLE 26 + +// SD-Card +#define MISO 40 +#define MOSI 38 +#define SCK 39 +#define SS 41 +#define SD_ENABLE 3 + +#define PIN_SD_MISO 40 +#define PIN_SD_MOSI 38 +#define PIN_SD_SCLK 39 +#define PIN_SD_CS 41 +#define PIN_SD_ENABLE 3 + +// USB +#define PIN_USB_DM 19 +#define PIN_USB_DP 20 + +// Camera +#define PWDN_GPIO_NUM 46 +#define RESET_GPIO_NUM -1 +#define XCLK_GPIO_NUM 15 +#define SIOD_GPIO_NUM 4 +#define SIOC_GPIO_NUM 5 + +#define Y9_GPIO_NUM 16 +#define Y8_GPIO_NUM 17 +#define Y7_GPIO_NUM 18 +#define Y6_GPIO_NUM 12 +#define Y5_GPIO_NUM 10 +#define Y4_GPIO_NUM 8 +#define Y3_GPIO_NUM 9 +#define Y2_GPIO_NUM 11 +#define VSYNC_GPIO_NUM 6 +#define HREF_GPIO_NUM 7 +#define PCLK_GPIO_NUM 13 + +// LoRa +#define LORA_TX 43 +#define LORA_RX 44 + +#endif /* Pins_Arduino_h */ \ No newline at end of file diff --git a/variants/sensebox_eye/tinyuf2.bin b/variants/sensebox_eye/tinyuf2.bin new file mode 100644 index 0000000000000000000000000000000000000000..6b0fb1bcd0417ce9175fc75bf09b49f7516ab078 GIT binary patch literal 186912 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 d9 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