mirror of
https://github.com/espressif/esp-matter.git
synced 2026-04-27 19:13:13 +00:00
177 lines
6.7 KiB
C++
177 lines
6.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 <esp_log.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
|
|
#include <device.h>
|
|
#include <esp_matter.h>
|
|
#include <esp_matter_console.h>
|
|
#include <led_driver.h>
|
|
|
|
#include <app_reset.h>
|
|
#include <app_priv.h>
|
|
|
|
using chip::kInvalidClusterId;
|
|
static constexpr chip::CommandId kInvalidCommandId = 0xFFFF'FFFF;
|
|
|
|
using namespace esp_matter;
|
|
using namespace esp_matter::cluster;
|
|
|
|
static const char *TAG = "app_driver";
|
|
extern uint16_t switch_endpoint_id;
|
|
static uint32_t g_cluster_id = kInvalidClusterId;
|
|
static uint32_t g_command_id = kInvalidCommandId;
|
|
|
|
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: <endpoint_id> <cluster_id> <command_id>. "
|
|
"Example: matter esp bound invoke 0x0001 0x0006 0x0002.\n");
|
|
} else if (argc == 4 && strncmp(argv[0], "invoke", sizeof("invoke")) == 0) {
|
|
uint16_t endpoint_id = strtol((const char *)&argv[1][2], NULL, 16);
|
|
uint32_t cluster_id = strtol((const char *)&argv[2][2], NULL, 16);
|
|
uint32_t command_id = strtol((const char *)&argv[3][2], NULL, 16);
|
|
|
|
g_cluster_id = cluster_id;
|
|
g_command_id = command_id;
|
|
client::cluster_update(endpoint_id, cluster_id);
|
|
} else {
|
|
ESP_LOGE(TAG, "Incorrect arguments. Check help for more details.");
|
|
return ESP_ERR_INVALID_ARG;
|
|
}
|
|
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>. "
|
|
"Example: matter esp client invoke 0x0001 0xBC5C01 0x0001 0x0006 0x0002.\n");
|
|
} else if (argc == 6 && strncmp(argv[0], "invoke", sizeof("invoke")) == 0) {
|
|
uint8_t fabric_index = strtol((const char *)&argv[1][2], NULL, 16);
|
|
uint64_t node_id = strtol((const char *)&argv[2][2], NULL, 16);
|
|
uint16_t remote_endpoint_id = strtol((const char *)&argv[3][2], NULL, 16);
|
|
uint32_t cluster_id = strtol((const char *)&argv[4][2], NULL, 16);
|
|
uint32_t command_id = strtol((const char *)&argv[5][2], NULL, 16);
|
|
|
|
g_cluster_id = cluster_id;
|
|
g_command_id = command_id;
|
|
client::connect(fabric_index, node_id, remote_endpoint_id);
|
|
} else {
|
|
ESP_LOGE(TAG, "Incorrect arguments. Check help for more details.");
|
|
return ESP_ERR_INVALID_ARG;
|
|
}
|
|
return ESP_OK;
|
|
}
|
|
|
|
static void app_driver_register_commands()
|
|
{
|
|
/* Add console command for bound devices */
|
|
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_command(&bound_command);
|
|
|
|
/* Add console command for client to control non-bound devices */
|
|
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_command(&client_command);
|
|
}
|
|
|
|
void app_driver_client_command_callback(client::peer_device_t *peer_device, uint16_t remote_endpoint_id,
|
|
void *priv_data)
|
|
{
|
|
/** TODO: Find a better way to get the cluster_id and command_id.
|
|
Once done, move the console commands to esp_matter_client. */
|
|
if (g_cluster_id == OnOff::Id) {
|
|
if (g_command_id == OnOff::Commands::Off::Id) {
|
|
on_off::command::send_off(peer_device, remote_endpoint_id);
|
|
} else if (g_command_id == OnOff::Commands::On::Id) {
|
|
on_off::command::send_on(peer_device, remote_endpoint_id);
|
|
} else if (g_command_id == OnOff::Commands::Toggle::Id) {
|
|
on_off::command::send_toggle(peer_device, remote_endpoint_id);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void app_driver_button_toggle_cb(void *arg)
|
|
{
|
|
ESP_LOGI(TAG, "Toggle button pressed");
|
|
uint16_t endpoint_id = switch_endpoint_id;
|
|
uint32_t cluster_id = OnOff::Id;
|
|
uint32_t command_id = OnOff::Commands::Toggle::Id;
|
|
|
|
g_cluster_id = cluster_id;
|
|
g_command_id = command_id;
|
|
client::cluster_update(endpoint_id, cluster_id);
|
|
}
|
|
|
|
esp_err_t app_driver_attribute_update(uint16_t endpoint_id, uint32_t cluster_id, uint32_t attribute_id,
|
|
esp_matter_attr_val_t *val)
|
|
{
|
|
/* Nothing to do here */
|
|
return ESP_OK;
|
|
}
|
|
|
|
static esp_err_t app_driver_attribute_set_defaults()
|
|
{
|
|
/* Get the default value (current value) from esp_matter and update the app_driver */
|
|
esp_err_t err = ESP_OK;
|
|
node_t *node = node::get();
|
|
endpoint_t *endpoint = endpoint::get_first(node);
|
|
while (endpoint) {
|
|
uint16_t endpoint_id = endpoint::get_id(endpoint);
|
|
cluster_t *cluster = cluster::get_first(endpoint);
|
|
while (cluster) {
|
|
uint32_t cluster_id = cluster::get_id(cluster);
|
|
attribute_t *attribute = attribute::get_first(cluster);
|
|
while (attribute) {
|
|
uint32_t attribute_id = attribute::get_id(attribute);
|
|
esp_matter_attr_val_t val = esp_matter_invalid(NULL);
|
|
err |= attribute::get_val(attribute, &val);
|
|
err |= app_driver_attribute_update(endpoint_id, cluster_id, attribute_id, &val);
|
|
attribute = attribute::get_next(attribute);
|
|
}
|
|
cluster = cluster::get_next(cluster);
|
|
}
|
|
endpoint = endpoint::get_next(endpoint);
|
|
}
|
|
return err;
|
|
}
|
|
|
|
esp_err_t app_driver_init()
|
|
{
|
|
ESP_LOGI(TAG, "Initialising driver");
|
|
|
|
/* Initialize button */
|
|
button_config_t button_config = button_driver_get_config();
|
|
button_handle_t handle = iot_button_create(&button_config);
|
|
iot_button_register_cb(handle, BUTTON_PRESS_DOWN, app_driver_button_toggle_cb);
|
|
app_reset_button_register(handle);
|
|
|
|
app_driver_attribute_set_defaults();
|
|
app_driver_register_commands();
|
|
client::set_command_callback(app_driver_client_command_callback, NULL);
|
|
return ESP_OK;
|
|
}
|