blob: 6ade0c4c7eb5dfa89b7128cc1ba23507370fec7b [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.
*/
/**
*
* Copyright (c) 2020 Silicon Labs
*
* 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.
*/
/*****************************************************************************
* @file
* @brief Application event code that is common to both the SOC and EZSP platforms.
*******************************************************************************
******************************************************************************/
#include <app/util/af-event.h>
#include <app/util/af.h>
#include <app/util/attribute-storage.h>
#include <platform/CHIPDeviceLayer.h>
#include <system/SystemTimer.h>
#define EMBER_MAX_EVENT_CONTROL_DELAY_MS (UINT32_MAX / 2)
#define EMBER_MAX_EVENT_CONTROL_DELAY_QS (EMBER_MAX_EVENT_CONTROL_DELAY_MS >> 8)
#define EMBER_MAX_EVENT_CONTROL_DELAY_MINUTES (EMBER_MAX_EVENT_CONTROL_DELAY_MS >> 16)
#include "gen/af-gen-event.h"
#include "gen/callback.h"
using namespace chip;
struct EmberEventData
{
/** The control structure for the event. */
EmberEventControl * control;
/** The procedure to call when the event fires. */
void (*handler)(void);
};
// *****************************************************************************
// Globals
#ifdef EMBER_AF_GENERATED_EVENT_CODE
// Stubs for IAS Zone Client Cluster issue #2057
EmberEventControl emberAfPluginIasZoneClientStateMachineEventControl;
void emberAfPluginIasZoneClientStateMachineEventHandler(void){};
EMBER_AF_GENERATED_EVENT_CODE
#endif // EMBER_AF_GENERATED_EVENT_CODE
#if defined(EMBER_AF_GENERATED_EVENT_CONTEXT)
uint16_t emAfAppEventContextLength = EMBER_AF_EVENT_CONTEXT_LENGTH;
EmberAfEventContext emAfAppEventContext[] = { EMBER_AF_GENERATED_EVENT_CONTEXT };
#endif // EMBER_AF_GENERATED_EVENT_CONTEXT
const char * emAfEventStrings[] = {
#ifdef EMBER_AF_GENERATED_EVENTS
EMBER_AF_GENERATED_EVENT_STRINGS
#endif
NULL,
};
EmberEventData emAfEvents[] = {
#ifdef EMBER_AF_GENERATED_EVENTS
EMBER_AF_GENERATED_EVENTS
#endif
{ NULL, NULL }
};
void EventControlHandler(chip::System::Layer * systemLayer, void * appState, chip::System::Error error)
{
EmberEventControl * control = reinterpret_cast<EmberEventControl *>(appState);
if (control->status != EMBER_EVENT_INACTIVE)
{
for (auto & event : emAfEvents)
{
if (event.control == control)
{
control->status = EMBER_EVENT_INACTIVE;
event.handler();
return;
}
}
}
}
const char emAfStackEventString[] = "Stack";
// *****************************************************************************
// Functions
// A function used to initialize events for idling
void emAfInitEvents(void) {}
const char * emberAfGetEventString(uint8_t index)
{
return (index == 0XFF ? emAfStackEventString : emAfEventStrings[index]);
}
static EmberAfEventContext * findEventContext(EndpointId endpoint, ClusterId clusterId, bool isClient)
{
#if defined(EMBER_AF_GENERATED_EVENT_CONTEXT)
uint16_t i;
for (i = 0; i < emAfAppEventContextLength; i++)
{
EmberAfEventContext * context = &(emAfAppEventContext[i]);
if (context->endpoint == endpoint && context->clusterId == clusterId && context->isClient == isClient)
{
return context;
}
}
#endif // EMBER_AF_GENERATED_EVENT_CONTEXT
return NULL;
}
EmberStatus emberEventControlSetDelayMS(EmberEventControl * control, uint32_t delayMs)
{
if (delayMs <= EMBER_MAX_EVENT_CONTROL_DELAY_MS)
{
control->status = EMBER_EVENT_MS_TIME;
#if !CHIP_DEVICE_LAYER_NONE
chip::DeviceLayer::SystemLayer.StartTimer(delayMs, EventControlHandler, control);
#endif
}
else
{
return EMBER_BAD_ARGUMENT;
}
return EMBER_SUCCESS;
}
void emberEventControlSetInactive(EmberEventControl * control)
{
if (control->status != EMBER_EVENT_INACTIVE)
{
control->status = EMBER_EVENT_INACTIVE;
#if !CHIP_DEVICE_LAYER_NONE
chip::DeviceLayer::SystemLayer.CancelTimer(EventControlHandler, control);
#endif
}
}
bool emberEventControlGetActive(EmberEventControl * control)
{
return control->status != EMBER_EVENT_INACTIVE;
}
void emberEventControlSetActive(EmberEventControl * control)
{
control->status = EMBER_EVENT_ZERO_DELAY;
#if !CHIP_DEVICE_LAYER_NONE
chip::DeviceLayer::SystemLayer.ScheduleWork(EventControlHandler, control);
#endif
}
EmberStatus emberAfEventControlSetDelayQS(EmberEventControl * control, uint32_t delayQs)
{
if (delayQs <= EMBER_MAX_EVENT_CONTROL_DELAY_QS)
{
return emberEventControlSetDelayMS(control, delayQs << 8);
}
else
{
return EMBER_BAD_ARGUMENT;
}
}
EmberStatus emberAfEventControlSetDelayMinutes(EmberEventControl * control, uint16_t delayM)
{
if (delayM <= EMBER_MAX_EVENT_CONTROL_DELAY_MINUTES)
{
return emberEventControlSetDelayMS(control, static_cast<uint32_t>(delayM) << 16);
}
else
{
return EMBER_BAD_ARGUMENT;
}
}
EmberStatus emberAfScheduleTickExtended(EndpointId endpoint, ClusterId clusterId, bool isClient, uint32_t delayMs,
EmberAfEventPollControl pollControl, EmberAfEventSleepControl sleepControl)
{
EmberAfEventContext * context = findEventContext(endpoint, clusterId, isClient);
// Disabled endpoints cannot schedule events. This will catch the problem in
// simulation.
EMBER_TEST_ASSERT(emberAfEndpointIsEnabled(endpoint));
if (context != NULL && emberAfEndpointIsEnabled(endpoint) &&
(emberEventControlSetDelayMS(context->eventControl, delayMs) == EMBER_SUCCESS))
{
context->pollControl = pollControl;
context->sleepControl = sleepControl;
return EMBER_SUCCESS;
}
return EMBER_BAD_ARGUMENT;
}
EmberStatus emberAfScheduleClusterTick(EndpointId endpoint, ClusterId clusterId, bool isClient, uint32_t delayMs,
EmberAfEventSleepControl sleepControl)
{
return emberAfScheduleTickExtended(endpoint, clusterId, isClient, delayMs,
(sleepControl == EMBER_AF_OK_TO_HIBERNATE ? EMBER_AF_LONG_POLL : EMBER_AF_SHORT_POLL),
(sleepControl == EMBER_AF_STAY_AWAKE ? EMBER_AF_STAY_AWAKE : EMBER_AF_OK_TO_SLEEP));
}
EmberStatus emberAfScheduleClientTickExtended(EndpointId endpoint, ClusterId clusterId, uint32_t delayMs,
EmberAfEventPollControl pollControl, EmberAfEventSleepControl sleepControl)
{
return emberAfScheduleTickExtended(endpoint, clusterId, EMBER_AF_CLIENT_CLUSTER_TICK, delayMs, pollControl, sleepControl);
}
EmberStatus emberAfScheduleClientTick(EndpointId endpoint, ClusterId clusterId, uint32_t delayMs)
{
return emberAfScheduleClientTickExtended(endpoint, clusterId, delayMs, EMBER_AF_LONG_POLL, EMBER_AF_OK_TO_SLEEP);
}
EmberStatus emberAfScheduleServerTickExtended(EndpointId endpoint, ClusterId clusterId, uint32_t delayMs,
EmberAfEventPollControl pollControl, EmberAfEventSleepControl sleepControl)
{
return emberAfScheduleTickExtended(endpoint, clusterId, EMBER_AF_SERVER_CLUSTER_TICK, delayMs, pollControl, sleepControl);
}
EmberStatus emberAfScheduleServerTick(EndpointId endpoint, ClusterId clusterId, uint32_t delayMs)
{
return emberAfScheduleServerTickExtended(endpoint, clusterId, delayMs, EMBER_AF_LONG_POLL, EMBER_AF_OK_TO_SLEEP);
}
EmberStatus emberAfDeactivateClusterTick(EndpointId endpoint, ClusterId clusterId, bool isClient)
{
EmberAfEventContext * context = findEventContext(endpoint, clusterId, isClient);
if (context != NULL)
{
emberEventControlSetInactive(context->eventControl);
return EMBER_SUCCESS;
}
return EMBER_BAD_ARGUMENT;
}
EmberStatus emberAfDeactivateClientTick(EndpointId endpoint, ClusterId clusterId)
{
return emberAfDeactivateClusterTick(endpoint, clusterId, EMBER_AF_CLIENT_CLUSTER_TICK);
}
EmberStatus emberAfDeactivateServerTick(EndpointId endpoint, ClusterId clusterId)
{
return emberAfDeactivateClusterTick(endpoint, clusterId, EMBER_AF_SERVER_CLUSTER_TICK);
}
#define MS_TO_QS(ms) ((ms) >> 8)
#define MS_TO_MIN(ms) ((ms) >> 16)
#define QS_TO_MS(qs) ((qs) << 8)
#define MIN_TO_MS(min) ((min) << 16)
// Used to calculate the duration and unit used by the host to set the sleep timer
void emAfGetTimerDurationAndUnitFromMS(uint32_t durationMs, uint16_t * duration, EmberEventUnits * units)
{
if (durationMs <= MAX_TIMER_UNITS_HOST)
{
*duration = (uint16_t) durationMs;
*units = EMBER_EVENT_MS_TIME;
}
else if (MS_TO_QS(durationMs) <= MAX_TIMER_UNITS_HOST)
{
*duration = (uint16_t)(MS_TO_QS(durationMs));
*units = EMBER_EVENT_QS_TIME;
}
else
{
*duration = (MS_TO_MIN(durationMs) <= MAX_TIMER_UNITS_HOST ? (uint16_t)(MS_TO_MIN(durationMs)) : MAX_TIMER_UNITS_HOST);
*units = EMBER_EVENT_MINUTE_TIME;
}
}
uint32_t emAfGetMSFromTimerDurationAndUnit(uint16_t duration, EmberEventUnits units)
{
uint32_t ms;
if (units == EMBER_EVENT_MS_TIME)
{
ms = duration;
}
else if (units == EMBER_EVENT_QS_TIME)
{
ms = QS_TO_MS(static_cast<uint32_t>(duration));
}
else if (units == EMBER_EVENT_MINUTE_TIME)
{
ms = MIN_TO_MS(static_cast<uint32_t>(duration));
}
else if (units == EMBER_EVENT_ZERO_DELAY)
{
ms = 0;
}
else
{
ms = UINT32_MAX;
}
return ms;
}