blob: 0165fdc437cd4065cc6f96ec618b997839975848 [file] [log] [blame]
/*
*
* Copyright (c) 2020 Project CHIP Authors
*
* 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 "BindingHandler.h"
#include "AppConfig.h"
#include "app/CommandSender.h"
#include "app/clusters/bindings/BindingManager.h"
#include "app/server/Server.h"
#include "controller/InvokeInteraction.h"
#include "platform/CHIPDeviceLayer.h"
#include <app/clusters/bindings/bindings.h>
#include <lib/support/CodeUtils.h>
using namespace chip;
using namespace chip::app;
using namespace chip::app::Clusters::LevelControl;
namespace {
void ProcessOnOffUnicastBindingCommand(CommandId commandId, const EmberBindingTableEntry & binding,
Messaging::ExchangeManager * exchangeMgr, const SessionHandle & sessionHandle)
{
auto onSuccess = [](const ConcreteCommandPath & commandPath, const StatusIB & status, const auto & dataResponse) {
ChipLogProgress(NotSpecified, "OnOff command succeeds");
};
auto onFailure = [](CHIP_ERROR error) {
ChipLogError(NotSpecified, "OnOff command failed: %" CHIP_ERROR_FORMAT, error.Format());
};
switch (commandId)
{
case Clusters::OnOff::Commands::Toggle::Id:
Clusters::OnOff::Commands::Toggle::Type toggleCommand;
Controller::InvokeCommandRequest(exchangeMgr, sessionHandle, binding.remote, toggleCommand, onSuccess, onFailure);
break;
case Clusters::OnOff::Commands::On::Id:
Clusters::OnOff::Commands::On::Type onCommand;
Controller::InvokeCommandRequest(exchangeMgr, sessionHandle, binding.remote, onCommand, onSuccess, onFailure);
break;
case Clusters::OnOff::Commands::Off::Id:
Clusters::OnOff::Commands::Off::Type offCommand;
Controller::InvokeCommandRequest(exchangeMgr, sessionHandle, binding.remote, offCommand, onSuccess, onFailure);
break;
}
}
void ProcessOnOffGroupBindingCommand(CommandId commandId, const EmberBindingTableEntry & binding)
{
Messaging::ExchangeManager & exchangeMgr = Server::GetInstance().GetExchangeManager();
switch (commandId)
{
case Clusters::OnOff::Commands::Toggle::Id:
Clusters::OnOff::Commands::Toggle::Type toggleCommand;
Controller::InvokeGroupCommandRequest(&exchangeMgr, binding.fabricIndex, binding.groupId, toggleCommand);
break;
case Clusters::OnOff::Commands::On::Id:
Clusters::OnOff::Commands::On::Type onCommand;
Controller::InvokeGroupCommandRequest(&exchangeMgr, binding.fabricIndex, binding.groupId, onCommand);
break;
case Clusters::OnOff::Commands::Off::Id:
Clusters::OnOff::Commands::Off::Type offCommand;
Controller::InvokeGroupCommandRequest(&exchangeMgr, binding.fabricIndex, binding.groupId, offCommand);
break;
}
}
void ProcessLevelControlUnicastBindingCommand(BindingCommandData * data, const EmberBindingTableEntry & binding,
OperationalDeviceProxy * peer_device)
{
auto onSuccess = [](const ConcreteCommandPath & commandPath, const StatusIB & status, const auto & dataResponse) {
ChipLogProgress(NotSpecified, "LevelControl command succeeds");
};
auto onFailure = [](CHIP_ERROR error) {
ChipLogError(NotSpecified, "LevelControl command failed: %" CHIP_ERROR_FORMAT, error.Format());
};
VerifyOrDie(peer_device != nullptr && peer_device->ConnectionReady());
switch (data->commandId)
{
case Clusters::LevelControl::Commands::MoveToLevel::Id: {
Clusters::LevelControl::Commands::MoveToLevel::Type moveToLevelCommand;
if (auto moveToLevel = std::get_if<BindingCommandData::MoveToLevel>(&data->commandData))
{
moveToLevelCommand.level = moveToLevel->level;
moveToLevelCommand.transitionTime = moveToLevel->transitionTime;
moveToLevelCommand.optionsMask = moveToLevel->optionsMask;
moveToLevelCommand.optionsOverride = moveToLevel->optionsOverride;
Controller::InvokeCommandRequest(peer_device->GetExchangeManager(), peer_device->GetSecureSession().Value(),
binding.remote, moveToLevelCommand, onSuccess, onFailure);
}
break;
}
case Clusters::LevelControl::Commands::Move::Id: {
Clusters::LevelControl::Commands::Move::Type moveCommand;
if (auto move = std::get_if<BindingCommandData::Move>(&data->commandData))
{
moveCommand.moveMode = move->moveMode;
moveCommand.rate = move->rate;
moveCommand.optionsMask = move->optionsMask;
moveCommand.optionsOverride = move->optionsOverride;
Controller::InvokeCommandRequest(peer_device->GetExchangeManager(), peer_device->GetSecureSession().Value(),
binding.remote, moveCommand, onSuccess, onFailure);
}
break;
}
case Clusters::LevelControl::Commands::Step::Id: {
Clusters::LevelControl::Commands::Step::Type stepCommand;
if (auto step = std::get_if<BindingCommandData::Step>(&data->commandData))
{
stepCommand.stepMode = step->stepMode;
stepCommand.stepSize = step->stepSize;
stepCommand.transitionTime = step->transitionTime;
stepCommand.optionsMask = step->optionsMask;
stepCommand.optionsOverride = step->optionsOverride;
Controller::InvokeCommandRequest(peer_device->GetExchangeManager(), peer_device->GetSecureSession().Value(),
binding.remote, stepCommand, onSuccess, onFailure);
}
break;
}
case Clusters::LevelControl::Commands::Stop::Id: {
Clusters::LevelControl::Commands::Stop::Type stopCommand;
if (auto stop = std::get_if<BindingCommandData::Stop>(&data->commandData))
{
stopCommand.optionsMask = stop->optionsMask;
stopCommand.optionsOverride = stop->optionsOverride;
Controller::InvokeCommandRequest(peer_device->GetExchangeManager(), peer_device->GetSecureSession().Value(),
binding.remote, stopCommand, onSuccess, onFailure);
}
break;
}
case Clusters::LevelControl::Commands::MoveToLevelWithOnOff::Id: {
Clusters::LevelControl::Commands::MoveToLevelWithOnOff::Type moveToLevelWithOnOffCommand;
if (auto moveToLevel = std::get_if<BindingCommandData::MoveToLevel>(&data->commandData))
{
moveToLevelWithOnOffCommand.level = moveToLevel->level;
moveToLevelWithOnOffCommand.transitionTime = moveToLevel->transitionTime;
moveToLevelWithOnOffCommand.optionsMask = moveToLevel->optionsMask;
moveToLevelWithOnOffCommand.optionsOverride = moveToLevel->optionsOverride;
Controller::InvokeCommandRequest(peer_device->GetExchangeManager(), peer_device->GetSecureSession().Value(),
binding.remote, moveToLevelWithOnOffCommand, onSuccess, onFailure);
}
break;
}
case Clusters::LevelControl::Commands::MoveWithOnOff::Id: {
Clusters::LevelControl::Commands::MoveWithOnOff::Type moveWithOnOffCommand;
if (auto move = std::get_if<BindingCommandData::Move>(&data->commandData))
{
moveWithOnOffCommand.moveMode = move->moveMode;
moveWithOnOffCommand.rate = move->rate;
moveWithOnOffCommand.optionsMask = move->optionsMask;
moveWithOnOffCommand.optionsOverride = move->optionsOverride;
Controller::InvokeCommandRequest(peer_device->GetExchangeManager(), peer_device->GetSecureSession().Value(),
binding.remote, moveWithOnOffCommand, onSuccess, onFailure);
}
break;
}
case Clusters::LevelControl::Commands::StepWithOnOff::Id: {
Clusters::LevelControl::Commands::StepWithOnOff::Type stepWithOnOffCommand;
if (auto step = std::get_if<BindingCommandData::Step>(&data->commandData))
{
stepWithOnOffCommand.stepMode = step->stepMode;
stepWithOnOffCommand.stepSize = step->stepSize;
stepWithOnOffCommand.transitionTime = step->transitionTime;
stepWithOnOffCommand.optionsMask = step->optionsMask;
stepWithOnOffCommand.optionsOverride = step->optionsOverride;
Controller::InvokeCommandRequest(peer_device->GetExchangeManager(), peer_device->GetSecureSession().Value(),
binding.remote, stepWithOnOffCommand, onSuccess, onFailure);
}
break;
}
case Clusters::LevelControl::Commands::StopWithOnOff::Id: {
Clusters::LevelControl::Commands::StopWithOnOff::Type stopWithOnOffCommand;
if (auto stop = std::get_if<BindingCommandData::Stop>(&data->commandData))
{
stopWithOnOffCommand.optionsMask = stop->optionsMask;
stopWithOnOffCommand.optionsOverride = stop->optionsOverride;
Controller::InvokeCommandRequest(peer_device->GetExchangeManager(), peer_device->GetSecureSession().Value(),
binding.remote, stopWithOnOffCommand, onSuccess, onFailure);
}
break;
}
default:
break;
}
}
void ProcessLevelControlGroupBindingCommand(BindingCommandData * data, const EmberBindingTableEntry & binding)
{
Messaging::ExchangeManager & exchangeMgr = Server::GetInstance().GetExchangeManager();
switch (data->commandId)
{
case Clusters::LevelControl::Commands::MoveToLevel::Id: {
Clusters::LevelControl::Commands::MoveToLevel::Type moveToLevelCommand;
if (auto moveToLevel = std::get_if<BindingCommandData::MoveToLevel>(&data->commandData))
{
moveToLevelCommand.level = moveToLevel->level;
moveToLevelCommand.transitionTime = moveToLevel->transitionTime;
moveToLevelCommand.optionsMask = moveToLevel->optionsMask;
moveToLevelCommand.optionsOverride = moveToLevel->optionsOverride;
Controller::InvokeGroupCommandRequest(&exchangeMgr, binding.fabricIndex, binding.groupId, moveToLevelCommand);
}
break;
}
case Clusters::LevelControl::Commands::Move::Id: {
Clusters::LevelControl::Commands::Move::Type moveCommand;
if (auto move = std::get_if<BindingCommandData::Move>(&data->commandData))
{
moveCommand.moveMode = move->moveMode;
moveCommand.rate = move->rate;
moveCommand.optionsMask = move->optionsMask;
moveCommand.optionsOverride = move->optionsOverride;
Controller::InvokeGroupCommandRequest(&exchangeMgr, binding.fabricIndex, binding.groupId, moveCommand);
}
break;
}
case Clusters::LevelControl::Commands::Step::Id: {
Clusters::LevelControl::Commands::Step::Type stepCommand;
if (auto step = std::get_if<BindingCommandData::Step>(&data->commandData))
{
stepCommand.stepMode = step->stepMode;
stepCommand.stepSize = step->stepSize;
stepCommand.transitionTime = step->transitionTime;
stepCommand.optionsMask = step->optionsMask;
stepCommand.optionsOverride = step->optionsOverride;
Controller::InvokeGroupCommandRequest(&exchangeMgr, binding.fabricIndex, binding.groupId, stepCommand);
}
break;
}
case Clusters::LevelControl::Commands::Stop::Id: {
Clusters::LevelControl::Commands::Stop::Type stopCommand;
if (auto stop = std::get_if<BindingCommandData::Stop>(&data->commandData))
{
stopCommand.optionsMask = stop->optionsMask;
stopCommand.optionsOverride = stop->optionsOverride;
Controller::InvokeGroupCommandRequest(&exchangeMgr, binding.fabricIndex, binding.groupId, stopCommand);
}
break;
}
case Clusters::LevelControl::Commands::MoveToLevelWithOnOff::Id: {
Clusters::LevelControl::Commands::MoveToLevelWithOnOff::Type moveToLevelWithOnOffCommand;
if (auto moveToLevel = std::get_if<BindingCommandData::MoveToLevel>(&data->commandData))
{
moveToLevelWithOnOffCommand.level = moveToLevel->level;
moveToLevelWithOnOffCommand.transitionTime = moveToLevel->transitionTime;
moveToLevelWithOnOffCommand.optionsMask = moveToLevel->optionsMask;
moveToLevelWithOnOffCommand.optionsOverride = moveToLevel->optionsOverride;
Controller::InvokeGroupCommandRequest(&exchangeMgr, binding.fabricIndex, binding.groupId, moveToLevelWithOnOffCommand);
}
break;
}
case Clusters::LevelControl::Commands::MoveWithOnOff::Id: {
Clusters::LevelControl::Commands::MoveWithOnOff::Type moveWithOnOffCommand;
if (auto move = std::get_if<BindingCommandData::Move>(&data->commandData))
{
moveWithOnOffCommand.moveMode = move->moveMode;
moveWithOnOffCommand.rate = move->rate;
moveWithOnOffCommand.optionsMask = move->optionsMask;
moveWithOnOffCommand.optionsOverride = move->optionsOverride;
Controller::InvokeGroupCommandRequest(&exchangeMgr, binding.fabricIndex, binding.groupId, moveWithOnOffCommand);
}
break;
}
case Clusters::LevelControl::Commands::StepWithOnOff::Id: {
Clusters::LevelControl::Commands::StepWithOnOff::Type stepWithOnOffCommand;
if (auto step = std::get_if<BindingCommandData::Step>(&data->commandData))
{
stepWithOnOffCommand.stepMode = step->stepMode;
stepWithOnOffCommand.stepSize = step->stepSize;
stepWithOnOffCommand.transitionTime = step->transitionTime;
stepWithOnOffCommand.optionsMask = step->optionsMask;
stepWithOnOffCommand.optionsOverride = step->optionsOverride;
Controller::InvokeGroupCommandRequest(&exchangeMgr, binding.fabricIndex, binding.groupId, stepWithOnOffCommand);
}
break;
}
case Clusters::LevelControl::Commands::StopWithOnOff::Id: {
Clusters::LevelControl::Commands::StopWithOnOff::Type stopWithOnOffCommand;
if (auto stop = std::get_if<BindingCommandData::Stop>(&data->commandData))
{
stopWithOnOffCommand.optionsMask = stop->optionsMask;
stopWithOnOffCommand.optionsOverride = stop->optionsOverride;
Controller::InvokeGroupCommandRequest(&exchangeMgr, binding.fabricIndex, binding.groupId, stopWithOnOffCommand);
}
break;
}
default:
break;
}
}
void LightSwitchChangedHandler(const EmberBindingTableEntry & binding, OperationalDeviceProxy * peer_device, void * context)
{
VerifyOrReturn(context != nullptr, ChipLogError(NotSpecified, "OnDeviceConnectedFn: context is null"));
BindingCommandData * data = static_cast<BindingCommandData *>(context);
if (binding.type == MATTER_MULTICAST_BINDING && data->isGroup)
{
switch (data->clusterId)
{
case Clusters::OnOff::Id:
ProcessOnOffGroupBindingCommand(data->commandId, binding);
break;
case Clusters::LevelControl::Id:
ProcessLevelControlGroupBindingCommand(data, binding);
break;
}
}
else if (binding.type == MATTER_UNICAST_BINDING && !data->isGroup)
{
switch (data->clusterId)
{
case Clusters::OnOff::Id:
VerifyOrDie(peer_device != nullptr && peer_device->ConnectionReady());
ProcessOnOffUnicastBindingCommand(data->commandId, binding, peer_device->GetExchangeManager(),
peer_device->GetSecureSession().Value());
break;
case Clusters::LevelControl::Id:
ProcessLevelControlUnicastBindingCommand(data, binding, peer_device);
break;
}
}
}
void LightSwitchContextReleaseHandler(void * context)
{
VerifyOrReturn(context != nullptr, ChipLogError(NotSpecified, "LightSwitchContextReleaseHandler: context is null"));
Platform::Delete(static_cast<BindingCommandData *>(context));
}
void InitBindingHandlerInternal(intptr_t arg)
{
auto & server = chip::Server::GetInstance();
chip::BindingManager::GetInstance().Init(
{ &server.GetFabricTable(), server.GetCASESessionManager(), &server.GetPersistentStorage() });
chip::BindingManager::GetInstance().RegisterBoundDeviceChangedHandler(LightSwitchChangedHandler);
chip::BindingManager::GetInstance().RegisterBoundDeviceContextReleaseHandler(LightSwitchContextReleaseHandler);
}
} // namespace
/********************************************************
* Switch functions
*********************************************************/
void SwitchWorkerFunction(intptr_t context)
{
VerifyOrReturn(context != 0, ChipLogError(NotSpecified, "SwitchWorkerFunction - Invalid work data"));
BindingCommandData * data = reinterpret_cast<BindingCommandData *>(context);
BindingManager::GetInstance().NotifyBoundClusterChanged(data->localEndpointId, data->clusterId, static_cast<void *>(data));
Platform::Delete(data);
}
void BindingWorkerFunction(intptr_t context)
{
VerifyOrReturn(context != 0, ChipLogError(NotSpecified, "BindingWorkerFunction - Invalid work data"));
EmberBindingTableEntry * entry = reinterpret_cast<EmberBindingTableEntry *>(context);
AddBindingEntry(*entry);
Platform::Delete(entry);
}
CHIP_ERROR InitBindingHandler()
{
// The initialization of binding manager will try establishing connection with unicast peers
// so it requires the Server instance to be correctly initialized. Post the init function to
// the event queue so that everything is ready when initialization is conducted.
chip::DeviceLayer::PlatformMgr().ScheduleWork(InitBindingHandlerInternal);
return CHIP_NO_ERROR;
}