Files
esp-matter/examples/sensors/main/app_main.cpp
T
Shubham Patil 73b457477d examples/sensors: Example to demonstrate few sensors
Added the temperature, humidity, and occupancy sensor on different
endpoints.

Fixes https://github.com/espressif/esp-matter/issues/1105
2024-11-11 11:25:56 +05:30

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));
}