mirror of
https://github.com/espressif/esp-matter.git
synced 2026-04-27 19:13:13 +00:00
768163a9da
- Driver specific changes for updated button component in examples. - Updated the device_hal code for button component upgrade. - Updated the idf_component.yml of examples to use latest version of espressif/cmake_utilities.
373 lines
16 KiB
C++
373 lines
16 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 <stdlib.h>
|
|
#include <string.h>
|
|
|
|
#include <esp_log.h>
|
|
#include <esp_matter.h>
|
|
#include <esp_matter_console.h>
|
|
|
|
#include <app_priv.h>
|
|
|
|
#include <app/server/Server.h>
|
|
#include "app/CommandPathParams.h"
|
|
#include <device.h>
|
|
|
|
using namespace chip::app::Clusters;
|
|
using namespace esp_matter;
|
|
using namespace esp_matter::cluster;
|
|
|
|
static const char *TAG = "app_driver";
|
|
extern uint16_t light_endpoint_id;
|
|
|
|
#if CONFIG_ENABLE_CHIP_SHELL
|
|
static char console_buffer[101] = {0};
|
|
static esp_err_t app_driver_bound_console_handler(int argc, char **argv)
|
|
{
|
|
if (argc == 1 && strncmp(argv[0], "help", sizeof("help")) == 0) {
|
|
printf("Bound commands:\n"
|
|
"\thelp: Print help\n"
|
|
"\tinvoke: <local_endpoint_id> <cluster_id> <command_id> parameters ... \n"
|
|
"\t\tExample: matter esp bound invoke 0x0001 0x0008 0x0000 0x50 0x0 0x1 0x1.\n");
|
|
} else if (argc >= 4 && strncmp(argv[0], "invoke", sizeof("invoke")) == 0) {
|
|
client::request_handle_t req_handle;
|
|
req_handle.type = esp_matter::client::INVOKE_CMD;
|
|
uint16_t local_endpoint_id = strtoul((const char *)&argv[1][2], NULL, 16);
|
|
req_handle.command_path.mClusterId = strtoul((const char *)&argv[2][2], NULL, 16);
|
|
req_handle.command_path.mCommandId = strtoul((const char *)&argv[3][2], NULL, 16);
|
|
|
|
if (argc > 4) {
|
|
console_buffer[0] = argc - 4;
|
|
for (int i = 0; i < (argc - 4); i++) {
|
|
if ((argv[4+i][0] != '0') || (argv[4+i][1] != 'x') || (strlen((const char*)&argv[4+i][2]) > 10)) {
|
|
ESP_LOGE(TAG, "Incorrect arguments. Check help for more details.");
|
|
return ESP_ERR_INVALID_ARG;
|
|
}
|
|
strcpy((console_buffer + 1 + 10*i), &argv[4+i][2]);
|
|
}
|
|
|
|
req_handle.request_data = console_buffer;
|
|
}
|
|
|
|
client::cluster_update(local_endpoint_id, &req_handle);
|
|
}
|
|
else {
|
|
ESP_LOGE(TAG, "Incorrect arguments. Check help for more details.");
|
|
return ESP_ERR_INVALID_ARG;
|
|
}
|
|
|
|
console_buffer[0] = 0;
|
|
|
|
return ESP_OK;
|
|
}
|
|
|
|
static esp_err_t app_driver_client_console_handler(int argc, char **argv)
|
|
{
|
|
if (argc == 1 && strncmp(argv[0], "help", sizeof("help")) == 0) {
|
|
printf("Client commands:\n"
|
|
"\thelp: Print help\n"
|
|
"\tinvoke: <fabric_index> <remote_node_id> <remote_endpoint_id> <cluster_id> <command_id> parameters ... \n"
|
|
"\t\tExample: matter esp client invoke 0x0001 0xBC5C01 0x0001 0x0008 0x0000 0x50 0x0 0x1 0x1.\n"
|
|
"\tinvoke-group: <fabric_index> <group_id> <cluster_id> <command_id> parameters ... \n"
|
|
"\t\tExample: matter esp client invoke-group 0x0001 0x257 0x0008 0x0000 0x50 0x0 0x1 0x1.\n");
|
|
} else if (argc >= 6 && strncmp(argv[0], "invoke", sizeof("invoke")) == 0) {
|
|
client::request_handle_t req_handle;
|
|
req_handle.type = esp_matter::client::INVOKE_CMD;
|
|
uint8_t fabric_index = strtoul((const char *)&argv[1][2], NULL, 16);
|
|
uint64_t node_id = strtoull((const char *)&argv[2][2], NULL, 16);
|
|
req_handle.command_path = {(chip::EndpointId)strtoul((const char *)&argv[3][2], NULL, 16) /* EndpointId */,
|
|
0 /* GroupId */, strtoul((const char *)&argv[4][2], NULL, 16) /* ClusterId */,
|
|
strtoul((const char *)&argv[5][2], NULL, 16) /* CommandId */,
|
|
chip::app::CommandPathFlags::kEndpointIdValid};
|
|
|
|
if (argc > 6) {
|
|
console_buffer[0] = argc - 6;
|
|
for (int i = 0; i < (argc - 6); i++) {
|
|
if ((argv[6+i][0] != '0') || (argv[6+i][1] != 'x') || (strlen((const char*)&argv[6+i][2]) > 10)) {
|
|
ESP_LOGE(TAG, "Incorrect arguments. Check help for more details.");
|
|
return ESP_ERR_INVALID_ARG;
|
|
}
|
|
strcpy((console_buffer + 1 + 10*i), &argv[6+i][2]);
|
|
}
|
|
|
|
req_handle.request_data = console_buffer;
|
|
}
|
|
|
|
auto &server = chip::Server::GetInstance();
|
|
client::connect(server.GetCASESessionManager(), fabric_index, node_id, &req_handle);
|
|
} else if (argc >= 5 && strncmp(argv[0], "invoke-group", sizeof("invoke-group")) == 0) {
|
|
client::request_handle_t req_handle;
|
|
req_handle.type = esp_matter::client::INVOKE_CMD;
|
|
uint8_t fabric_index = strtoul((const char *)&argv[1][2], NULL, 16);
|
|
req_handle.command_path = {
|
|
0 /* EndpointId */, (chip::GroupId)strtoul((const char *)&argv[2][2], NULL, 16) /* GroupId */,
|
|
strtoul((const char *)&argv[3][2], NULL, 16) /* ClusterId */,
|
|
strtoul((const char *)&argv[4][2], NULL, 16) /* CommandId */, chip::app::CommandPathFlags::kGroupIdValid};
|
|
|
|
if (argc > 5) {
|
|
console_buffer[0] = argc - 5;
|
|
for (int i = 0; i < (argc - 5); i++) {
|
|
if ((argv[5+i][0] != '0') || (argv[5+i][1] != 'x') || (strlen((const char*)&argv[5+i][2]) > 10)) {
|
|
ESP_LOGE(TAG, "Incorrect arguments. Check help for more details.");
|
|
return ESP_ERR_INVALID_ARG;
|
|
}
|
|
strcpy((console_buffer + 1 + 10*i), &argv[5+i][2]);
|
|
}
|
|
|
|
req_handle.request_data = console_buffer;
|
|
}
|
|
|
|
client::group_request_send(fabric_index, &req_handle);
|
|
}else {
|
|
ESP_LOGE(TAG, "Incorrect arguments. Check help for more details.");
|
|
return ESP_ERR_INVALID_ARG;
|
|
}
|
|
|
|
console_buffer[0] = 0;
|
|
|
|
return ESP_OK;
|
|
}
|
|
|
|
static void app_driver_register_commands()
|
|
{
|
|
/* Add console command for bound devices */
|
|
static const esp_matter::console::command_t bound_command = {
|
|
.name = "bound",
|
|
.description = "This can be used to simulate on-device control for bound devices."
|
|
"Usage: matter esp bound <bound_command>. "
|
|
"Bound commands: help, invoke",
|
|
.handler = app_driver_bound_console_handler,
|
|
};
|
|
esp_matter::console::add_commands(&bound_command, 1);
|
|
|
|
/* Add console command for client to control non-bound devices */
|
|
static const esp_matter::console::command_t client_command = {
|
|
.name = "client",
|
|
.description = "This can be used to simulate on-device control for client devices."
|
|
"Usage: matter esp client <client_command>. "
|
|
"Client commands: help, invoke",
|
|
.handler = app_driver_client_console_handler,
|
|
};
|
|
esp_matter::console::add_commands(&client_command, 1);
|
|
}
|
|
#endif // CONFIG_ENABLE_CHIP_SHELL
|
|
|
|
static void send_command_success_callback(void *context, const ConcreteCommandPath &command_path,
|
|
const chip::app::StatusIB &status, TLVReader *response_data)
|
|
{
|
|
ESP_LOGI(TAG, "Send command success");
|
|
}
|
|
|
|
static void send_command_failure_callback(void *context, CHIP_ERROR error)
|
|
{
|
|
ESP_LOGI(TAG, "Send command failure: err :%" CHIP_ERROR_FORMAT, error.Format());
|
|
}
|
|
|
|
void app_driver_client_invoke_command_callback(client::peer_device_t *peer_device, client::request_handle_t *req_handle,
|
|
void *priv_data)
|
|
{
|
|
if (req_handle->type != esp_matter::client::INVOKE_CMD ||
|
|
!req_handle->command_path.mFlags.Has(chip::app::CommandPathFlags::kEndpointIdValid)) {
|
|
return;
|
|
}
|
|
char command_data_str[32];
|
|
if (req_handle->command_path.mClusterId == OnOff::Id) {
|
|
strcpy(command_data_str, "{}");
|
|
} else if (req_handle->command_path.mClusterId == Identify::Id) {
|
|
if (req_handle->command_path.mCommandId == Identify::Commands::Identify::Id) {
|
|
if (((char *)req_handle->request_data)[0] != 1) {
|
|
ESP_LOGE(TAG, "Number of parameters error");
|
|
return;
|
|
}
|
|
sprintf(command_data_str, "{\"0:U16\": %ld}",
|
|
strtoul((const char *)(req_handle->request_data) + 1, NULL, 16));
|
|
} else {
|
|
ESP_LOGE(TAG, "Unsupported command");
|
|
return;
|
|
}
|
|
} else {
|
|
ESP_LOGE(TAG, "Unsupported cluster");
|
|
return;
|
|
}
|
|
client::interaction::invoke::send_request(NULL, peer_device, req_handle->command_path, command_data_str,
|
|
send_command_success_callback, send_command_failure_callback,
|
|
chip::NullOptional);
|
|
}
|
|
|
|
void app_driver_client_group_invoke_command_callback(uint8_t fabric_index, client::request_handle_t *req_handle, void *priv_data)
|
|
{
|
|
if (req_handle->type != esp_matter::client::INVOKE_CMD ||
|
|
!req_handle->command_path.mFlags.Has(chip::app::CommandPathFlags::kGroupIdValid)) {
|
|
return;
|
|
}
|
|
char command_data_str[32];
|
|
if (req_handle->command_path.mClusterId == OnOff::Id) {
|
|
strcpy(command_data_str, "{}");
|
|
} else if (req_handle->command_path.mClusterId == Identify::Id) {
|
|
if (req_handle->command_path.mCommandId == Identify::Commands::Identify::Id) {
|
|
if (((char *)req_handle->request_data)[0] != 1) {
|
|
ESP_LOGE(TAG, "Number of parameters error");
|
|
return;
|
|
}
|
|
sprintf(command_data_str, "{\"0:U16\": %ld}",
|
|
strtoul((const char *)(req_handle->request_data) + 1, NULL, 16));
|
|
} else {
|
|
ESP_LOGE(TAG, "Unsupported command");
|
|
return;
|
|
}
|
|
} else {
|
|
ESP_LOGE(TAG, "Unsupported cluster");
|
|
return;
|
|
}
|
|
client::interaction::invoke::send_group_request(fabric_index, req_handle->command_path, command_data_str);
|
|
}
|
|
|
|
/* Do any conversions/remapping for the actual value here */
|
|
static esp_err_t app_driver_light_set_power(led_driver_handle_t handle, esp_matter_attr_val_t *val)
|
|
{
|
|
return led_driver_set_power(handle, val->val.b);
|
|
}
|
|
|
|
static esp_err_t app_driver_light_set_brightness(led_driver_handle_t handle, esp_matter_attr_val_t *val)
|
|
{
|
|
int value = REMAP_TO_RANGE(val->val.u8, MATTER_BRIGHTNESS, STANDARD_BRIGHTNESS);
|
|
return led_driver_set_brightness(handle, value);
|
|
}
|
|
|
|
static esp_err_t app_driver_light_set_hue(led_driver_handle_t handle, esp_matter_attr_val_t *val)
|
|
{
|
|
int value = REMAP_TO_RANGE(val->val.u8, MATTER_HUE, STANDARD_HUE);
|
|
return led_driver_set_hue(handle, value);
|
|
}
|
|
|
|
static esp_err_t app_driver_light_set_saturation(led_driver_handle_t handle, esp_matter_attr_val_t *val)
|
|
{
|
|
int value = REMAP_TO_RANGE(val->val.u8, MATTER_SATURATION, STANDARD_SATURATION);
|
|
return led_driver_set_saturation(handle, value);
|
|
}
|
|
|
|
static esp_err_t app_driver_light_set_temperature(led_driver_handle_t handle, esp_matter_attr_val_t *val)
|
|
{
|
|
uint32_t value = REMAP_TO_RANGE_INVERSE(val->val.u16, STANDARD_TEMPERATURE_FACTOR);
|
|
return led_driver_set_temperature(handle, value);
|
|
}
|
|
|
|
static void app_driver_button_toggle_cb(void *arg, void *data)
|
|
{
|
|
ESP_LOGI(TAG, "Toggle button pressed");
|
|
uint16_t endpoint_id = light_endpoint_id;
|
|
uint32_t cluster_id = OnOff::Id;
|
|
uint32_t attribute_id = OnOff::Attributes::OnOff::Id;
|
|
|
|
attribute_t *attribute = attribute::get(endpoint_id, cluster_id, attribute_id);
|
|
|
|
esp_matter_attr_val_t val = esp_matter_invalid(NULL);
|
|
attribute::get_val(attribute, &val);
|
|
val.val.b = !val.val.b;
|
|
attribute::update(endpoint_id, cluster_id, attribute_id, &val);
|
|
}
|
|
|
|
esp_err_t app_driver_attribute_update(app_driver_handle_t driver_handle, uint16_t endpoint_id, uint32_t cluster_id,
|
|
uint32_t attribute_id, esp_matter_attr_val_t *val)
|
|
{
|
|
esp_err_t err = ESP_OK;
|
|
if (endpoint_id == light_endpoint_id) {
|
|
led_driver_handle_t handle = (led_driver_handle_t)driver_handle;
|
|
if (cluster_id == OnOff::Id) {
|
|
if (attribute_id == OnOff::Attributes::OnOff::Id) {
|
|
err = app_driver_light_set_power(handle, val);
|
|
}
|
|
} else if (cluster_id == LevelControl::Id) {
|
|
if (attribute_id == LevelControl::Attributes::CurrentLevel::Id) {
|
|
err = app_driver_light_set_brightness(handle, val);
|
|
}
|
|
} else if (cluster_id == ColorControl::Id) {
|
|
if (attribute_id == ColorControl::Attributes::CurrentHue::Id) {
|
|
err = app_driver_light_set_hue(handle, val);
|
|
} else if (attribute_id == ColorControl::Attributes::CurrentSaturation::Id) {
|
|
err = app_driver_light_set_saturation(handle, val);
|
|
} else if (attribute_id == ColorControl::Attributes::ColorTemperatureMireds::Id) {
|
|
err = app_driver_light_set_temperature(handle, val);
|
|
}
|
|
}
|
|
}
|
|
return err;
|
|
}
|
|
|
|
esp_err_t app_driver_light_set_defaults(uint16_t endpoint_id)
|
|
{
|
|
esp_err_t err = ESP_OK;
|
|
void *priv_data = endpoint::get_priv_data(endpoint_id);
|
|
led_driver_handle_t handle = (led_driver_handle_t)priv_data;
|
|
esp_matter_attr_val_t val = esp_matter_invalid(NULL);
|
|
|
|
/* Setting brightness */
|
|
attribute_t *attribute = attribute::get(endpoint_id, LevelControl::Id, LevelControl::Attributes::CurrentLevel::Id);
|
|
attribute::get_val(attribute, &val);
|
|
err |= app_driver_light_set_brightness(handle, &val);
|
|
|
|
/* Setting color */
|
|
attribute = attribute::get(endpoint_id, ColorControl::Id, ColorControl::Attributes::ColorMode::Id);
|
|
attribute::get_val(attribute, &val);
|
|
if (val.val.u8 == (uint8_t)ColorControl::ColorMode::kCurrentHueAndCurrentSaturation) {
|
|
/* Setting hue */
|
|
attribute = attribute::get(endpoint_id, ColorControl::Id, ColorControl::Attributes::CurrentHue::Id);
|
|
attribute::get_val(attribute, &val);
|
|
err |= app_driver_light_set_hue(handle, &val);
|
|
/* Setting saturation */
|
|
attribute = attribute::get(endpoint_id, ColorControl::Id, ColorControl::Attributes::CurrentSaturation::Id);
|
|
attribute::get_val(attribute, &val);
|
|
err |= app_driver_light_set_saturation(handle, &val);
|
|
} else if (val.val.u8 == (uint8_t)ColorControl::ColorMode::kColorTemperature) {
|
|
/* Setting temperature */
|
|
attribute = attribute::get(endpoint_id, ColorControl::Id, ColorControl::Attributes::ColorTemperatureMireds::Id);
|
|
attribute::get_val(attribute, &val);
|
|
err |= app_driver_light_set_temperature(handle, &val);
|
|
} else {
|
|
ESP_LOGE(TAG, "Color mode not supported");
|
|
}
|
|
|
|
/* Setting power */
|
|
attribute = attribute::get(endpoint_id, OnOff::Id, OnOff::Attributes::OnOff::Id);
|
|
attribute::get_val(attribute, &val);
|
|
err |= app_driver_light_set_power(handle, &val);
|
|
|
|
return err;
|
|
}
|
|
|
|
app_driver_handle_t app_driver_light_init()
|
|
{
|
|
/* Initialize led */
|
|
led_driver_config_t config = led_driver_get_config();
|
|
led_driver_handle_t handle = led_driver_init(&config);
|
|
return (app_driver_handle_t)handle;
|
|
}
|
|
|
|
app_driver_handle_t app_driver_button_init()
|
|
{
|
|
/* Initialize button */
|
|
button_handle_t handle = NULL;
|
|
const button_config_t btn_cfg = {0};
|
|
const button_gpio_config_t btn_gpio_cfg = button_driver_get_config();
|
|
|
|
if (iot_button_new_gpio_device(&btn_cfg, &btn_gpio_cfg, &handle) != ESP_OK) {
|
|
ESP_LOGE(TAG, "Failed to create button device");
|
|
return NULL;
|
|
}
|
|
|
|
iot_button_register_cb(handle, BUTTON_PRESS_DOWN, NULL, app_driver_button_toggle_cb, NULL);
|
|
|
|
#if CONFIG_ENABLE_CHIP_SHELL
|
|
app_driver_register_commands();
|
|
#endif // CONFIG_ENABLE_CHIP_SHELL
|
|
client::set_request_callback(app_driver_client_invoke_command_callback, app_driver_client_group_invoke_command_callback, NULL);
|
|
|
|
return (app_driver_handle_t)handle;
|
|
}
|