/* 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 #include #include #include #include #include #include #include #include #include #include #include static const char *TAG = "app_main"; uint16_t light_endpoint_id = 0; uint16_t aggregator_endpoint_id = 0; using namespace esp_matter; using namespace esp_matter::attribute; using namespace esp_matter::endpoint; using namespace chip::app::Clusters; static void app_event_cb(const ChipDeviceEvent *event, intptr_t arg) { switch (event->Type) { case chip::DeviceLayer::DeviceEventType::kInterfaceIpAddressChanged: ESP_LOGI(TAG, "Interface IP Address changed"); break; 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::kCommissioningSessionStarted: ESP_LOGI(TAG, "Commissioning session started"); break; case chip::DeviceLayer::DeviceEventType::kCommissioningSessionStopped: ESP_LOGI(TAG, "Commissioning session stopped"); break; case chip::DeviceLayer::DeviceEventType::kCommissioningWindowOpened: ESP_LOGI(TAG, "Commissioning window opened"); break; case chip::DeviceLayer::DeviceEventType::kCommissioningWindowClosed: ESP_LOGI(TAG, "Commissioning window closed"); 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: %d, effect: %d", type, effect_id); 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) { esp_err_t err = ESP_OK; if (type == PRE_UPDATE) { /* Driver update */ app_driver_handle_t driver_handle = (app_driver_handle_t)priv_data; err = app_driver_attribute_update(driver_handle, endpoint_id, cluster_id, attribute_id, val); } return err; } // This callback is invoked after the creation or resumption of a bridge endpoint. // It can be used to add data model elements (e.g., attributes, commands, etc.) to the bridge endpoint. esp_err_t create_bridge_devices(esp_matter::endpoint_t *ep, uint32_t device_type_id, void *priv_data) { esp_err_t err = ESP_OK; switch (device_type_id) { case ESP_MATTER_ON_OFF_LIGHT_DEVICE_TYPE_ID: { on_off_light::config_t on_off_light_conf; err = on_off_light::add(ep, &on_off_light_conf); break; } case ESP_MATTER_DIMMABLE_LIGHT_DEVICE_TYPE_ID: { dimmable_light::config_t dimmable_light_conf; err = dimmable_light::add(ep, &dimmable_light_conf); break; } case ESP_MATTER_COLOR_TEMPERATURE_LIGHT_DEVICE_TYPE_ID: { color_temperature_light::config_t color_temperature_light_conf; err = color_temperature_light::add(ep, &color_temperature_light_conf); break; } case ESP_MATTER_EXTENDED_COLOR_LIGHT_DEVICE_TYPE_ID: { extended_color_light::config_t extended_color_light_conf; err = extended_color_light::add(ep, &extended_color_light_conf); break; } case ESP_MATTER_ON_OFF_SWITCH_DEVICE_TYPE_ID: { on_off_switch::config_t switch_config; err = on_off_switch::add(ep, &switch_config); break; } default: { ESP_LOGE(TAG, "Unsupported bridged matter device type"); return ESP_ERR_INVALID_ARG; } } return err; } extern "C" void app_main() { esp_err_t err = ESP_OK; /* Initialize the ESP NVS layer */ nvs_flash_init(); /* Initialize driver */ app_driver_handle_t light_handle = app_driver_light_init(); app_driver_handle_t button_handle = app_driver_button_init(); app_reset_button_register(button_handle); /* Create a Matter node and add the mandatory Root Node device type on endpoint 0 */ node::config_t node_config; // node handle can be used to add/modify other endpoints. 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")); color_temperature_light::config_t light_config; light_config.on_off.on_off = DEFAULT_POWER; light_config.on_off_lighting.start_up_on_off = nullptr; light_config.level_control.current_level = DEFAULT_BRIGHTNESS; light_config.level_control_lighting.start_up_current_level = DEFAULT_BRIGHTNESS; light_config.color_control.color_mode = (uint8_t)ColorControl::ColorMode::kColorTemperature; light_config.color_control.enhanced_color_mode = (uint8_t)ColorControl::ColorMode::kColorTemperature; light_config.color_control_color_temperature.startup_color_temperature_mireds = nullptr; // endpoint handles can be used to add/modify clusters. endpoint_t *endpoint = color_temperature_light::create(node, &light_config, ENDPOINT_FLAG_NONE, light_handle); ABORT_APP_ON_FAILURE(endpoint != nullptr, ESP_LOGE(TAG, "Failed to create color temperature light endpoint")); aggregator::config_t aggregator_config; endpoint_t *aggregator = endpoint::aggregator::create(node, &aggregator_config, ENDPOINT_FLAG_NONE, NULL); ABORT_APP_ON_FAILURE(aggregator != nullptr, ESP_LOGE(TAG, "Failed to create aggregator endpoint")); light_endpoint_id = endpoint::get_id(endpoint); ESP_LOGI(TAG, "Light created with endpoint_id %d", light_endpoint_id); aggregator_endpoint_id = endpoint::get_id(aggregator); ESP_LOGI(TAG, "Aggregator created with endpoint id %d", aggregator_endpoint_id); /* Add additional features to the node */ cluster_t *cluster = cluster::get(endpoint, ColorControl::Id); cluster::color_control::feature::hue_saturation::config_t hue_saturation_config; hue_saturation_config.current_hue = DEFAULT_HUE; hue_saturation_config.current_saturation = DEFAULT_SATURATION; cluster::color_control::feature::hue_saturation::add(cluster, &hue_saturation_config); /* 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)); err = app_bridge_initialize(node, create_bridge_devices); ABORT_APP_ON_FAILURE(err == ESP_OK, ESP_LOGE(TAG, "Failed to resume the bridged endpoints: %d", err)); app_espnow_init(); /* Starting driver with default values */ app_driver_light_set_defaults(light_endpoint_id); #if CONFIG_ENABLE_CHIP_SHELL esp_matter::console::diagnostics_register_commands(); esp_matter::console::wifi_register_commands(); esp_matter::console::factoryreset_register_commands(); esp_matter::console::init(); #endif }