mirror of
https://github.com/espressif/esp-matter.git
synced 2026-04-28 11:28:32 +00:00
73b457477d
Added the temperature, humidity, and occupancy sensor on different endpoints. Fixes https://github.com/espressif/esp-matter/issues/1105
227 lines
9.7 KiB
C++
227 lines
9.7 KiB
C++
/*
|
|
This example code is in the Public Domain (or CC0 licensed, at your option.)
|
|
|
|
Unless required by applicable law or agreed to in writing, this
|
|
software is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
|
|
CONDITIONS OF ANY KIND, either express or implied.
|
|
*/
|
|
|
|
#include <app/server/CommissioningWindowManager.h>
|
|
#include <app/server/Server.h>
|
|
#include <bsp/esp-bsp.h>
|
|
#include <esp_err.h>
|
|
#include <esp_log.h>
|
|
#include <esp_matter.h>
|
|
#include <esp_matter_ota.h>
|
|
#include <nvs_flash.h>
|
|
|
|
#include <app_openthread_config.h>
|
|
#include <app_reset.h>
|
|
#include <common_macros.h>
|
|
|
|
// drivers implemented by this example
|
|
#include <drivers/shtc3.h>
|
|
#include <drivers/pir.h>
|
|
|
|
static const char *TAG = "app_main";
|
|
|
|
using namespace esp_matter;
|
|
using namespace esp_matter::attribute;
|
|
using namespace esp_matter::endpoint;
|
|
using namespace chip::app::Clusters;
|
|
|
|
// Application cluster specification, 7.18.2.11. Temperature
|
|
// represents a temperature on the Celsius scale with a resolution of 0.01°C.
|
|
// temp = (temperature in °C) x 100
|
|
static void temp_sensor_notification(uint16_t endpoint_id, float temp, void *user_data)
|
|
{
|
|
// schedule the attribute update so that we can report it from matter thread
|
|
chip::DeviceLayer::SystemLayer().ScheduleLambda([endpoint_id, temp]() {
|
|
attribute_t * attribute = attribute::get(endpoint_id,
|
|
TemperatureMeasurement::Id,
|
|
TemperatureMeasurement::Attributes::MeasuredValue::Id);
|
|
|
|
esp_matter_attr_val_t val = esp_matter_invalid(NULL);
|
|
attribute::get_val(attribute, &val);
|
|
val.val.i16 = static_cast<int16_t>(temp * 100);
|
|
|
|
attribute::update(endpoint_id, TemperatureMeasurement::Id, TemperatureMeasurement::Attributes::MeasuredValue::Id, &val);
|
|
});
|
|
}
|
|
|
|
// Application cluster specification, 2.6.4.1. MeasuredValue Attribute
|
|
// represents the humidity in percent.
|
|
// humidity = (humidity in %) x 100
|
|
static void humidity_sensor_notification(uint16_t endpoint_id, float humidity, void *user_data)
|
|
{
|
|
// schedule the attribute update so that we can report it from matter thread
|
|
chip::DeviceLayer::SystemLayer().ScheduleLambda([endpoint_id, humidity]() {
|
|
attribute_t * attribute = attribute::get(endpoint_id,
|
|
RelativeHumidityMeasurement::Id,
|
|
RelativeHumidityMeasurement::Attributes::MeasuredValue::Id);
|
|
|
|
esp_matter_attr_val_t val = esp_matter_invalid(NULL);
|
|
attribute::get_val(attribute, &val);
|
|
val.val.u16 = static_cast<uint16_t>(humidity * 100);
|
|
|
|
attribute::update(endpoint_id, RelativeHumidityMeasurement::Id, RelativeHumidityMeasurement::Attributes::MeasuredValue::Id, &val);
|
|
});
|
|
}
|
|
|
|
static void occupancy_sensor_notification(uint16_t endpoint_id, bool occupancy, void *user_data)
|
|
{
|
|
// schedule the attribute update so that we can report it from matter thread
|
|
chip::DeviceLayer::SystemLayer().ScheduleLambda([endpoint_id, occupancy]() {
|
|
attribute_t * attribute = attribute::get(endpoint_id,
|
|
OccupancySensing::Id,
|
|
OccupancySensing::Attributes::Occupancy::Id);
|
|
|
|
esp_matter_attr_val_t val = esp_matter_invalid(NULL);
|
|
attribute::get_val(attribute, &val);
|
|
val.val.b = occupancy;
|
|
|
|
attribute::update(endpoint_id, OccupancySensing::Id, OccupancySensing::Attributes::Occupancy::Id, &val);
|
|
});
|
|
}
|
|
|
|
static esp_err_t factory_reset_button_register()
|
|
{
|
|
button_handle_t push_button;
|
|
esp_err_t err = bsp_iot_button_create(&push_button, NULL, BSP_BUTTON_NUM);
|
|
VerifyOrReturnError(err == ESP_OK, err);
|
|
return app_reset_button_register(push_button);
|
|
}
|
|
|
|
static void open_commissioning_window_if_necessary()
|
|
{
|
|
VerifyOrReturn(chip::Server::GetInstance().GetFabricTable().FabricCount() == 0);
|
|
|
|
chip::CommissioningWindowManager & commissionMgr = chip::Server::GetInstance().GetCommissioningWindowManager();
|
|
VerifyOrReturn(commissionMgr.IsCommissioningWindowOpen() == false);
|
|
|
|
// After removing last fabric, this example does not remove the Wi-Fi credentials
|
|
// and still has IP connectivity so, only advertising on DNS-SD.
|
|
CHIP_ERROR err = commissionMgr.OpenBasicCommissioningWindow(chip::System::Clock::Seconds16(300),
|
|
chip::CommissioningWindowAdvertisement::kDnssdOnly);
|
|
if (err != CHIP_NO_ERROR)
|
|
{
|
|
ESP_LOGE(TAG, "Failed to open commissioning window, err:%" CHIP_ERROR_FORMAT, err.Format());
|
|
}
|
|
}
|
|
|
|
static void app_event_cb(const ChipDeviceEvent *event, intptr_t arg)
|
|
{
|
|
switch (event->Type) {
|
|
case chip::DeviceLayer::DeviceEventType::kCommissioningComplete:
|
|
ESP_LOGI(TAG, "Commissioning complete");
|
|
break;
|
|
|
|
case chip::DeviceLayer::DeviceEventType::kFailSafeTimerExpired:
|
|
ESP_LOGI(TAG, "Commissioning failed, fail safe timer expired");
|
|
break;
|
|
|
|
case chip::DeviceLayer::DeviceEventType::kFabricRemoved:
|
|
ESP_LOGI(TAG, "Fabric removed successfully");
|
|
open_commissioning_window_if_necessary();
|
|
break;
|
|
|
|
case chip::DeviceLayer::DeviceEventType::kBLEDeinitialized:
|
|
ESP_LOGI(TAG, "BLE deinitialized and memory reclaimed");
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
// This callback is invoked when clients interact with the Identify Cluster.
|
|
// In the callback implementation, an endpoint can identify itself. (e.g., by flashing an LED or light).
|
|
static esp_err_t app_identification_cb(identification::callback_type_t type, uint16_t endpoint_id, uint8_t effect_id,
|
|
uint8_t effect_variant, void *priv_data)
|
|
{
|
|
ESP_LOGI(TAG, "Identification callback: type: %u, effect: %u, variant: %u", type, effect_id, effect_variant);
|
|
return ESP_OK;
|
|
}
|
|
|
|
// 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
|
|
// is not of your interest, please do not return an error code and strictly return ESP_OK.
|
|
static esp_err_t app_attribute_update_cb(attribute::callback_type_t type, uint16_t endpoint_id, uint32_t cluster_id,
|
|
uint32_t attribute_id, esp_matter_attr_val_t *val, void *priv_data)
|
|
{
|
|
// Since this is just a sensor and we don't expect any writes on our temperature sensor,
|
|
// so, return success.
|
|
return ESP_OK;
|
|
}
|
|
|
|
extern "C" void app_main()
|
|
{
|
|
/* Initialize the ESP NVS layer */
|
|
nvs_flash_init();
|
|
|
|
/* Initialize push button on the dev-kit to reset the device */
|
|
esp_err_t err = factory_reset_button_register();
|
|
ABORT_APP_ON_FAILURE(ESP_OK == err, ESP_LOGE(TAG, "Failed to initialize reset button, err:%d", err));
|
|
|
|
/* Create a Matter node and add the mandatory Root Node device type on endpoint 0 */
|
|
node::config_t node_config;
|
|
node_t *node = node::create(&node_config, app_attribute_update_cb, app_identification_cb);
|
|
ABORT_APP_ON_FAILURE(node != nullptr, ESP_LOGE(TAG, "Failed to create Matter node"));
|
|
|
|
// add temperature sensor device
|
|
temperature_sensor::config_t temp_sensor_config;
|
|
endpoint_t * temp_sensor_ep = temperature_sensor::create(node, &temp_sensor_config, ENDPOINT_FLAG_NONE, NULL);
|
|
ABORT_APP_ON_FAILURE(temp_sensor_ep != nullptr, ESP_LOGE(TAG, "Failed to create temperature_sensor endpoint"));
|
|
|
|
// add the humidity sensor device
|
|
humidity_sensor::config_t humidity_sensor_config;
|
|
endpoint_t * humidity_sensor_ep = humidity_sensor::create(node, &humidity_sensor_config, ENDPOINT_FLAG_NONE, NULL);
|
|
ABORT_APP_ON_FAILURE(humidity_sensor_ep != nullptr, ESP_LOGE(TAG, "Failed to create humidity_sensor endpoint"));
|
|
|
|
// initialize temperature and humidity sensor driver (shtc3)
|
|
static shtc3_sensor_config_t shtc3_config = {
|
|
.temperature = {
|
|
.cb = temp_sensor_notification,
|
|
.endpoint_id = endpoint::get_id(temp_sensor_ep),
|
|
},
|
|
.humidity = {
|
|
.cb = humidity_sensor_notification,
|
|
.endpoint_id = endpoint::get_id(humidity_sensor_ep),
|
|
},
|
|
};
|
|
err = shtc3_sensor_init(&shtc3_config);
|
|
ABORT_APP_ON_FAILURE(err == ESP_OK, ESP_LOGE(TAG, "Failed to initialize temperature sensor driver"));
|
|
|
|
// add the occupancy sensor
|
|
occupancy_sensor::config_t occupancy_sensor_config;
|
|
occupancy_sensor_config.occupancy_sensing.occupancy_sensor_type =
|
|
chip::to_underlying(OccupancySensing::OccupancySensorTypeEnum::kPir);
|
|
occupancy_sensor_config.occupancy_sensing.occupancy_sensor_type_bitmap =
|
|
chip::to_underlying(OccupancySensing::OccupancySensorTypeBitmap::kPir);
|
|
|
|
endpoint_t * occupancy_sensor_ep = occupancy_sensor::create(node, &occupancy_sensor_config, ENDPOINT_FLAG_NONE, NULL);
|
|
ABORT_APP_ON_FAILURE(occupancy_sensor_ep != nullptr, ESP_LOGE(TAG, "Failed to create occupancy_sensor endpoint"));
|
|
|
|
// initialize occupancy sensor driver (pir)
|
|
static pir_sensor_config_t pir_config = {
|
|
.cb = occupancy_sensor_notification,
|
|
.endpoint_id = endpoint::get_id(occupancy_sensor_ep),
|
|
};
|
|
err = pir_sensor_init(&pir_config);
|
|
ABORT_APP_ON_FAILURE(err == ESP_OK, ESP_LOGE(TAG, "Failed to initialize occupancy sensor driver"));
|
|
|
|
#if CHIP_DEVICE_CONFIG_ENABLE_THREAD
|
|
/* Set OpenThread platform config */
|
|
esp_openthread_platform_config_t config = {
|
|
.radio_config = ESP_OPENTHREAD_DEFAULT_RADIO_CONFIG(),
|
|
.host_config = ESP_OPENTHREAD_DEFAULT_HOST_CONFIG(),
|
|
.port_config = ESP_OPENTHREAD_DEFAULT_PORT_CONFIG(),
|
|
};
|
|
set_openthread_platform_config(&config);
|
|
#endif
|
|
|
|
/* Matter start */
|
|
err = esp_matter::start(app_event_cb);
|
|
ABORT_APP_ON_FAILURE(err == ESP_OK, ESP_LOGE(TAG, "Failed to start Matter, err:%d", err));
|
|
}
|