Skip to content

Commit 46f2e2f

Browse files
committed
Merge branch 'add_door_lock_example' into 'main'
add door lock example See merge request app-frameworks/esp-matter!687
2 parents 2010527 + 6432697 commit 46f2e2f

24 files changed

+1928
-0
lines changed

components/esp_matter/esp_matter_feature.cpp

+1
Original file line numberDiff line numberDiff line change
@@ -4165,6 +4165,7 @@ esp_err_t add(cluster_t *cluster, config_t *config)
41654165
ESP_LOGE(TAG, "Should add at least one of PIN, RID, FGP and FACE feature before add USR feature");
41664166
return ESP_FAIL;
41674167
}
4168+
41684169
update_feature_map(cluster, get_id());
41694170

41704171
/* Attributes not managed internally */

components/esp_matter/esp_matter_feature.h

+1
Original file line numberDiff line numberDiff line change
@@ -1822,6 +1822,7 @@ esp_err_t add(cluster_t *cluster);
18221822

18231823
} /* feature */
18241824
} /* electrical_energy_measurement */
1825+
18251826
namespace door_lock {
18261827
namespace feature {
18271828

examples/.build-rules.yml

+6
Original file line numberDiff line numberDiff line change
@@ -92,3 +92,9 @@ examples/managed_component_light:
9292
- if: IDF_TARGET in [""]
9393
temporary: true
9494
reason: this should be compiled without setting up environment, another CI has been added
95+
96+
examples/door_lock:
97+
enable:
98+
- if: IDF_TARGET in ["esp32", "esp32c3", "esp32c2", "esp32c6", "esp32h2"]
99+
temporary: true
100+
reason: the other targets are not tested yet

examples/door_lock/CMakeLists.txt

+40
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,40 @@
1+
# The following lines of boilerplate have to be in your project's
2+
# CMakeLists in this exact order for cmake to work correctly
3+
cmake_minimum_required(VERSION 3.5)
4+
5+
if(NOT DEFINED ENV{ESP_MATTER_PATH})
6+
message(FATAL_ERROR "Please set ESP_MATTER_PATH to the path of esp-matter repo")
7+
endif(NOT DEFINED ENV{ESP_MATTER_PATH})
8+
9+
set(PROJECT_VER "1.0")
10+
set(PROJECT_VER_NUMBER 1)
11+
12+
set(ESP_MATTER_PATH $ENV{ESP_MATTER_PATH})
13+
set(MATTER_SDK_PATH ${ESP_MATTER_PATH}/connectedhomeip/connectedhomeip)
14+
15+
# This should be done before using the IDF_TARGET variable.
16+
include($ENV{IDF_PATH}/tools/cmake/project.cmake)
17+
18+
set(EXTRA_COMPONENT_DIRS
19+
"${ESP_MATTER_PATH}/examples/common"
20+
"${MATTER_SDK_PATH}/config/esp32/components"
21+
"${ESP_MATTER_PATH}/components"
22+
${extra_components_dirs_append})
23+
24+
project(door_lock)
25+
26+
# WARNING: This is just an example for using key for decrypting the encrypted OTA image
27+
# Please do not use it as is.
28+
if(CONFIG_ENABLE_ENCRYPTED_OTA)
29+
target_add_binary_data(light.elf "esp_image_encryption_key.pem" TEXT)
30+
endif()
31+
32+
if(CONFIG_IDF_TARGET_ESP32C2)
33+
include(relinker)
34+
endif()
35+
36+
idf_build_set_property(CXX_COMPILE_OPTIONS "-std=gnu++17;-Os;-DCHIP_HAVE_CONFIG_H;-Wno-overloaded-virtual" APPEND)
37+
idf_build_set_property(C_COMPILE_OPTIONS "-Os" APPEND)
38+
# For RISCV chips, project_include.cmake sets -Wno-format, but does not clear various
39+
# flags that depend on -Wformat
40+
idf_build_set_property(COMPILE_OPTIONS "-Wno-format-nonliteral;-Wno-format-security" APPEND)

examples/door_lock/README.md

+15
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
# Doorlock
2+
3+
This example creates a Doorlock device using the ESP Matter data model.
4+
5+
See the [docs](https://docs.espressif.com/projects/esp-matter/en/latest/esp32/developing.html) for more information about building and flashing the firmware.
6+
7+
## 1. Additional Environment Setup
8+
9+
No additional setup is required.
10+
11+
## 2. Post Commissioning Setup
12+
13+
No additional setup is required.
14+
15+
## 3. Device Performance
+6
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
idf_component_register(SRC_DIRS "." "./lock"
2+
PRIV_INCLUDE_DIRS
3+
"." "./lock" "${ESP_MATTER_PATH}/examples/common/utils")
4+
5+
set_property(TARGET ${COMPONENT_LIB} PROPERTY CXX_STANDARD 17)
6+
target_compile_options(${COMPONENT_LIB} PRIVATE "-DCHIP_HAVE_CONFIG_H")
+37
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,37 @@
1+
/*
2+
This example code is in the Public Domain (or CC0 licensed, at your option.)
3+
4+
Unless required by applicable law or agreed to in writing, this
5+
software is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
6+
CONDITIONS OF ANY KIND, either express or implied.
7+
*/
8+
9+
#include <esp_log.h>
10+
#include <stdlib.h>
11+
#include <string.h>
12+
13+
#include <esp_matter.h>
14+
#include "bsp/esp-bsp.h"
15+
16+
#include <app_priv.h>
17+
18+
static const char *TAG = "app_driver";
19+
20+
using namespace chip::app::Clusters;
21+
using namespace esp_matter;
22+
23+
esp_err_t app_driver_attribute_update(app_driver_handle_t driver_handle, uint16_t endpoint_id, uint32_t cluster_id,
24+
uint32_t attribute_id, esp_matter_attr_val_t *val)
25+
{
26+
esp_err_t err = ESP_OK;
27+
return err;
28+
}
29+
30+
app_driver_handle_t app_driver_button_init()
31+
{
32+
/* Initialize button */
33+
button_handle_t btns[BSP_BUTTON_NUM];
34+
ESP_ERROR_CHECK(bsp_iot_button_create(btns, NULL, BSP_BUTTON_NUM));
35+
36+
return (app_driver_handle_t)btns[0];
37+
}

examples/door_lock/main/app_main.cpp

+220
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,220 @@
1+
/*
2+
This example code is in the Public Domain (or CC0 licensed, at your option.)
3+
4+
Unless required by applicable law or agreed to in writing, this
5+
software is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
6+
CONDITIONS OF ANY KIND, either express or implied.
7+
*/
8+
9+
#include <esp_err.h>
10+
#include <esp_log.h>
11+
#include <nvs_flash.h>
12+
#if CONFIG_PM_ENABLE
13+
#include <esp_pm.h>
14+
#endif
15+
16+
#include <esp_matter.h>
17+
#include <esp_matter_console.h>
18+
#include <esp_matter_ota.h>
19+
20+
#include <common_macros.h>
21+
#include <app_priv.h>
22+
#if CHIP_DEVICE_CONFIG_ENABLE_THREAD
23+
#include <platform/ESP32/OpenthreadLauncher.h>
24+
#endif
25+
26+
#include <app/server/CommissioningWindowManager.h>
27+
#include <app/server/Server.h>
28+
29+
static const char *TAG = "app_main";
30+
uint16_t door_lock_endpoint_id = 0;
31+
32+
using namespace esp_matter;
33+
using namespace esp_matter::attribute;
34+
using namespace esp_matter::endpoint;
35+
using namespace chip::app::Clusters;
36+
using namespace chip;
37+
38+
constexpr auto k_timeout_seconds = 300;
39+
40+
#if CONFIG_ENABLE_ENCRYPTED_OTA
41+
extern const char decryption_key_start[] asm("_binary_esp_image_encryption_key_pem_start");
42+
extern const char decryption_key_end[] asm("_binary_esp_image_encryption_key_pem_end");
43+
44+
static const char *s_decryption_key = decryption_key_start;
45+
static const uint16_t s_decryption_key_len = decryption_key_end - decryption_key_start;
46+
#endif // CONFIG_ENABLE_ENCRYPTED_OTA
47+
48+
static void app_event_cb(const ChipDeviceEvent *event, intptr_t arg)
49+
{
50+
switch (event->Type) {
51+
case chip::DeviceLayer::DeviceEventType::kInterfaceIpAddressChanged:
52+
ESP_LOGI(TAG, "Interface IP Address changed");
53+
break;
54+
55+
case chip::DeviceLayer::DeviceEventType::kCommissioningComplete:
56+
ESP_LOGI(TAG, "Commissioning complete");
57+
break;
58+
59+
case chip::DeviceLayer::DeviceEventType::kFailSafeTimerExpired:
60+
ESP_LOGI(TAG, "Commissioning failed, fail safe timer expired");
61+
break;
62+
63+
case chip::DeviceLayer::DeviceEventType::kCommissioningSessionStarted:
64+
ESP_LOGI(TAG, "Commissioning session started");
65+
break;
66+
67+
case chip::DeviceLayer::DeviceEventType::kCommissioningSessionStopped:
68+
ESP_LOGI(TAG, "Commissioning session stopped");
69+
break;
70+
71+
case chip::DeviceLayer::DeviceEventType::kCommissioningWindowOpened:
72+
ESP_LOGI(TAG, "Commissioning window opened");
73+
break;
74+
75+
case chip::DeviceLayer::DeviceEventType::kCommissioningWindowClosed:
76+
ESP_LOGI(TAG, "Commissioning window closed");
77+
break;
78+
79+
case chip::DeviceLayer::DeviceEventType::kFabricRemoved:
80+
{
81+
ESP_LOGI(TAG, "Fabric removed successfully");
82+
if (chip::Server::GetInstance().GetFabricTable().FabricCount() == 0)
83+
{
84+
chip::CommissioningWindowManager & commissionMgr = chip::Server::GetInstance().GetCommissioningWindowManager();
85+
constexpr auto kTimeoutSeconds = chip::System::Clock::Seconds16(k_timeout_seconds);
86+
if (!commissionMgr.IsCommissioningWindowOpen())
87+
{
88+
/* After removing last fabric, this example does not remove the Wi-Fi credentials
89+
* and still has IP connectivity so, only advertising on DNS-SD.
90+
*/
91+
CHIP_ERROR err = commissionMgr.OpenBasicCommissioningWindow(kTimeoutSeconds,
92+
chip::CommissioningWindowAdvertisement::kDnssdOnly);
93+
if (err != CHIP_NO_ERROR)
94+
{
95+
ESP_LOGE(TAG, "Failed to open commissioning window, err:%" CHIP_ERROR_FORMAT, err.Format());
96+
}
97+
}
98+
}
99+
break;
100+
}
101+
102+
case chip::DeviceLayer::DeviceEventType::kFabricWillBeRemoved:
103+
ESP_LOGI(TAG, "Fabric will be removed");
104+
break;
105+
106+
case chip::DeviceLayer::DeviceEventType::kFabricUpdated:
107+
ESP_LOGI(TAG, "Fabric is updated");
108+
break;
109+
110+
case chip::DeviceLayer::DeviceEventType::kFabricCommitted:
111+
ESP_LOGI(TAG, "Fabric is committed");
112+
break;
113+
114+
case chip::DeviceLayer::DeviceEventType::kBLEDeinitialized:
115+
ESP_LOGI(TAG, "BLE deinitialized and memory reclaimed");
116+
break;
117+
118+
default:
119+
break;
120+
}
121+
}
122+
123+
// This callback is invoked when clients interact with the Identify Cluster.
124+
// In the callback implementation, an endpoint can identify itself. (e.g., by flashing an LED or light).
125+
static esp_err_t app_identification_cb(identification::callback_type_t type, uint16_t endpoint_id, uint8_t effect_id,
126+
uint8_t effect_variant, void *priv_data)
127+
{
128+
ESP_LOGI(TAG, "Identification callback: type: %u, effect: %u, variant: %u", type, effect_id, effect_variant);
129+
return ESP_OK;
130+
}
131+
132+
// This callback is called for every attribute update. The callback implementation shall
133+
// handle the desired attributes and return an appropriate error code. If the attribute
134+
// is not of your interest, please do not return an error code and strictly return ESP_OK.
135+
static esp_err_t app_attribute_update_cb(attribute::callback_type_t type, uint16_t endpoint_id, uint32_t cluster_id,
136+
uint32_t attribute_id, esp_matter_attr_val_t *val, void *priv_data)
137+
{
138+
esp_err_t err = ESP_OK;
139+
140+
if (type == PRE_UPDATE) {
141+
/* Driver update */
142+
app_driver_handle_t driver_handle = (app_driver_handle_t)priv_data;
143+
err = app_driver_attribute_update(driver_handle, endpoint_id, cluster_id, attribute_id, val);
144+
}
145+
146+
return err;
147+
}
148+
149+
extern "C" void app_main()
150+
{
151+
esp_err_t err = ESP_OK;
152+
153+
/* Initialize the ESP NVS layer */
154+
nvs_flash_init();
155+
156+
#if CONFIG_PM_ENABLE
157+
esp_pm_config_t pm_config = {
158+
.max_freq_mhz = CONFIG_ESP_DEFAULT_CPU_FREQ_MHZ,
159+
.min_freq_mhz = CONFIG_ESP_DEFAULT_CPU_FREQ_MHZ,
160+
#if CONFIG_FREERTOS_USE_TICKLESS_IDLE
161+
.light_sleep_enable = true
162+
#endif
163+
};
164+
err = esp_pm_configure(&pm_config);
165+
#endif
166+
167+
/* Create a Matter node and add the mandatory Root Node device type on endpoint 0 */
168+
node::config_t node_config;
169+
170+
// node handle can be used to add/modify other endpoints.
171+
node_t *node = node::create(&node_config, app_attribute_update_cb, app_identification_cb);
172+
ABORT_APP_ON_FAILURE(node != nullptr, ESP_LOGE(TAG, "Failed to create Matter node"));
173+
174+
door_lock::config_t door_lock_config;
175+
cluster::door_lock::feature::credential_over_the_air_access::config_t cota_config;
176+
cluster::door_lock::feature::pin_credential::config_t pin_credential_config;
177+
cluster::door_lock::feature::user::config_t user_config;
178+
// endpoint handles can be used to add/modify clusters.
179+
endpoint_t *endpoint = door_lock::create(node, &door_lock_config, ENDPOINT_FLAG_NONE, NULL);
180+
ABORT_APP_ON_FAILURE(endpoint != nullptr, ESP_LOGE(TAG, "Failed to create door lock endpoint"));
181+
cluster_t *door_lock_cluster = cluster::get(endpoint, DoorLock::Id);
182+
cluster::door_lock::feature::credential_over_the_air_access::add(door_lock_cluster, &cota_config);
183+
cluster::door_lock::feature::pin_credential::add(door_lock_cluster, &pin_credential_config);
184+
cluster::door_lock::feature::user::add(door_lock_cluster, &user_config);
185+
cluster::door_lock::attribute::create_auto_relock_time(door_lock_cluster, 5);
186+
187+
door_lock_endpoint_id = endpoint::get_id(endpoint);
188+
ESP_LOGI(TAG, "Door lock created with endpoint_id %d", door_lock_endpoint_id);
189+
190+
#if CHIP_DEVICE_CONFIG_ENABLE_THREAD
191+
/* Set OpenThread platform config */
192+
esp_openthread_platform_config_t config = {
193+
.radio_config = ESP_OPENTHREAD_DEFAULT_RADIO_CONFIG(),
194+
.host_config = ESP_OPENTHREAD_DEFAULT_HOST_CONFIG(),
195+
.port_config = ESP_OPENTHREAD_DEFAULT_PORT_CONFIG(),
196+
};
197+
set_openthread_platform_config(&config);
198+
#endif
199+
200+
/* Matter start */
201+
err = esp_matter::start(app_event_cb);
202+
ABORT_APP_ON_FAILURE(err == ESP_OK, ESP_LOGE(TAG, "Failed to start Matter, err:%d", err));
203+
204+
/* do nothing now */
205+
door_lock_init();
206+
207+
#if CONFIG_ENABLE_ENCRYPTED_OTA
208+
err = esp_matter_ota_requestor_encrypted_init(s_decryption_key, s_decryption_key_len);
209+
ABORT_APP_ON_FAILURE(err == ESP_OK, ESP_LOGE(TAG, "Failed to initialized the encrypted OTA, err: %d", err));
210+
#endif // CONFIG_ENABLE_ENCRYPTED_OTA
211+
212+
#if CONFIG_ENABLE_CHIP_SHELL
213+
esp_matter::console::diagnostics_register_commands();
214+
esp_matter::console::wifi_register_commands();
215+
#if CONFIG_OPENTHREAD_CLI
216+
esp_matter::console::otcli_register_commands();
217+
#endif
218+
esp_matter::console::init();
219+
#endif
220+
}

0 commit comments

Comments
 (0)