| /* |
| * |
| * Copyright (c) 2025 Project CHIP Authors |
| * All rights reserved. |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| #include "CodegenIntegration.h" |
| #include <app-common/zap-generated/attributes/Accessors.h> |
| #include <app-common/zap-generated/cluster-objects.h> |
| #include <app/clusters/occupancy-sensor-server/OccupancySensingCluster.h> |
| #include <app/server-cluster/OptionalAttributeSet.h> |
| #include <app/static-cluster-config/OccupancySensing.h> |
| #include <app/util/attribute-storage.h> |
| #include <app/util/endpoint-config-api.h> |
| #include <data-model-providers/codegen/ClusterIntegration.h> |
| #include <data-model-providers/codegen/CodegenDataModelProvider.h> |
| #include <platform/DefaultTimerDelegate.h> |
| |
| using namespace chip; |
| using namespace chip::app; |
| using namespace chip::app::Clusters; |
| using namespace chip::app::Clusters::OccupancySensing; |
| using namespace chip::app::Clusters::OccupancySensing::Attributes; |
| |
| namespace { |
| |
| constexpr size_t kOccupancySensingFixedClusterCount = OccupancySensing::StaticApplicationConfig::kFixedClusterConfig.size(); |
| constexpr size_t kOccupancySensingMaxClusterCount = kOccupancySensingFixedClusterCount + CHIP_DEVICE_CONFIG_DYNAMIC_ENDPOINT_COUNT; |
| |
| LazyRegisteredServerCluster<OccupancySensingCluster> gServers[kOccupancySensingMaxClusterCount]; |
| |
| DefaultTimerDelegate gDefaultTimerDelegate; |
| |
| class IntegrationDelegate : public CodegenClusterIntegration::Delegate |
| { |
| public: |
| ServerClusterRegistration & CreateRegistration(EndpointId endpointId, unsigned clusterInstanceIndex, |
| uint32_t optionalAttributeBits, uint32_t featureMap) override |
| { |
| OccupancySensingCluster::Config config(endpointId); |
| |
| // No features enabled (defaults to PIR). If the app needs other features, it MUST instantiate and configure the cluster |
| // directly instead of relying on CodegenIntegration. |
| config.WithFeatures(BitFlags<Feature>(0u)); |
| |
| // If the optional HoldTime attribute is enabled, enable the HoldTime logic. |
| // The delay attributes are required if the corresponding sensor feature is present. |
| if (AttributeSet(optionalAttributeBits).IsSet(Attributes::HoldTime::Id)) |
| { |
| // Initializes hold time with default limits and timer delegate. The Application can use SetHoldTimeLimits() and |
| // SetHoldTime() later to customize. Initial defaults come from typical values found in real devices on the market. |
| constexpr chip::app::Clusters::OccupancySensing::Structs::HoldTimeLimitsStruct::Type kDefaultHoldTimeLimits = { |
| .holdTimeMin = 1, .holdTimeMax = 300, .holdTimeDefault = 30 |
| }; |
| config.WithHoldTime(kDefaultHoldTimeLimits.holdTimeDefault, kDefaultHoldTimeLimits, gDefaultTimerDelegate); |
| |
| // Show deprecated attributes if enabled in Zap |
| config.WithDeprecatedAttributes( |
| emberAfContainsAttribute(endpointId, OccupancySensing::Id, Attributes::PIROccupiedToUnoccupiedDelay::Id) || |
| emberAfContainsAttribute(endpointId, OccupancySensing::Id, Attributes::UltrasonicOccupiedToUnoccupiedDelay::Id) || |
| emberAfContainsAttribute(endpointId, OccupancySensing::Id, |
| Attributes::PhysicalContactOccupiedToUnoccupiedDelay::Id)); |
| } |
| |
| gServers[clusterInstanceIndex].Create(config); |
| return gServers[clusterInstanceIndex].Registration(); |
| } |
| |
| ServerClusterInterface * FindRegistration(unsigned clusterInstanceIndex) override |
| { |
| VerifyOrReturnValue(gServers[clusterInstanceIndex].IsConstructed(), nullptr); |
| return &gServers[clusterInstanceIndex].Cluster(); |
| } |
| |
| void ReleaseRegistration(unsigned clusterInstanceIndex) override { gServers[clusterInstanceIndex].Destroy(); } |
| }; |
| |
| } // namespace |
| |
| void MatterOccupancySensingClusterInitCallback(EndpointId endpointId) |
| { |
| IntegrationDelegate integrationDelegate; |
| |
| CodegenClusterIntegration::RegisterServer( |
| { |
| .endpointId = endpointId, |
| .clusterId = OccupancySensing::Id, |
| .fixedClusterInstanceCount = kOccupancySensingFixedClusterCount, |
| .maxClusterInstanceCount = kOccupancySensingMaxClusterCount, |
| .fetchFeatureMap = false, |
| .fetchOptionalAttributes = true, |
| }, |
| integrationDelegate); |
| } |
| |
| void MatterOccupancySensingClusterShutdownCallback(EndpointId endpointId, MatterClusterShutdownType shutdownType) |
| { |
| IntegrationDelegate integrationDelegate; |
| |
| CodegenClusterIntegration::UnregisterServer( |
| { |
| .endpointId = endpointId, |
| .clusterId = OccupancySensing::Id, |
| .fixedClusterInstanceCount = kOccupancySensingFixedClusterCount, |
| .maxClusterInstanceCount = kOccupancySensingMaxClusterCount, |
| }, |
| integrationDelegate, shutdownType); |
| } |
| |
| namespace chip::app::Clusters::OccupancySensing { |
| |
| OccupancySensingCluster * FindClusterOnEndpoint(EndpointId endpointId) |
| { |
| IntegrationDelegate integrationDelegate; |
| |
| ServerClusterInterface * occupancySensing = CodegenClusterIntegration::FindClusterOnEndpoint( |
| { |
| .endpointId = endpointId, |
| .clusterId = OccupancySensing::Id, |
| .fixedClusterInstanceCount = kOccupancySensingFixedClusterCount, |
| .maxClusterInstanceCount = kOccupancySensingMaxClusterCount, |
| }, |
| integrationDelegate); |
| |
| return static_cast<OccupancySensingCluster *>(occupancySensing); |
| } |
| |
| } // namespace chip::app::Clusters::OccupancySensing |
| |
| // Legacy PluginServer callback stubs |
| void MatterOccupancySensingPluginServerInitCallback() {} |
| void MatterOccupancySensingPluginServerShutdownCallback() {} |