blob: a420c7de069ac5b30e16ce908f6e52f0a52aae55 [file] [log] [blame]
/*
*
* Copyright (c) 2021 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.
*/
/**
* @file
* This file implements WatchableEvents using libevent.
*/
#include <platform/CHIPDeviceBuildConfig.h>
#include <support/CodeUtils.h>
#include <system/SystemLayer.h>
#include <system/SystemSockets.h>
#if CHIP_DEVICE_CONFIG_ENABLE_MDNS && !__ZEPHYR__
namespace chip {
namespace Mdns {
void GetMdnsTimeout(timeval & timeout);
void HandleMdnsTimeout();
} // namespace Mdns
} // namespace chip
#endif // CHIP_DEVICE_CONFIG_ENABLE_MDNS && !__ZEPHYR__
#ifndef CHIP_CONFIG_LIBEVENT_DEBUG_CHECKS
#define CHIP_CONFIG_LIBEVENT_DEBUG_CHECKS 1 // TODO(#5556): default to off
#endif
namespace chip {
namespace System {
namespace {
System::SocketEvents SocketEventsFromLibeventFlags(short eventFlags)
{
return System::SocketEvents()
.Set(SocketEventFlags::kRead, eventFlags & EV_READ)
.Set(SocketEventFlags::kWrite, eventFlags & EV_WRITE);
}
void TimeoutCallbackHandler(evutil_socket_t fd, short eventFlags, void * data)
{
event * const ev = reinterpret_cast<event *>(data);
evtimer_del(ev);
}
} // anonymous namespace
void WatchableEventManager::Init(System::Layer & systemLayer)
{
#if CHIP_CONFIG_LIBEVENT_DEBUG_CHECKS
static bool enabled_event_debug_mode = false;
if (!enabled_event_debug_mode)
{
enabled_event_debug_mode = true;
event_enable_debug_mode();
}
#endif // CHIP_CONFIG_LIBEVENT_DEBUG_CHECKS
mEventBase = event_base_new();
mTimeoutEvent = evtimer_new(mEventBase, TimeoutCallbackHandler, event_self_cbarg());
mActiveSockets = nullptr;
mSystemLayer = &systemLayer;
}
void WatchableEventManager::PrepareEvents()
{
// TODO(#5556): Integrate timer platform details with WatchableEventManager.
timeval nextTimeout = { 0, 0 };
PrepareEventsWithTimeout(nextTimeout);
}
void WatchableEventManager::PrepareEventsWithTimeout(struct timeval & nextTimeout)
{
// TODO(#5556): Integrate timer platform details with WatchableEventManager.
mSystemLayer->GetTimeout(nextTimeout);
if (nextTimeout.tv_sec || nextTimeout.tv_usec)
{
evtimer_add(mTimeoutEvent, &nextTimeout);
}
}
void WatchableEventManager::WaitForEvents()
{
VerifyOrDie(mEventBase != nullptr);
event_base_loop(mEventBase, EVLOOP_ONCE);
}
void WatchableEventManager::HandleEvents()
{
mSystemLayer->HandleTimeout();
#if CHIP_DEVICE_CONFIG_ENABLE_MDNS && !__ZEPHYR__
chip::Mdns::HandleMdnsTimeout();
#endif // CHIP_DEVICE_CONFIG_ENABLE_MDNS && !__ZEPHYR__
while (mActiveSockets != nullptr)
{
WatchableSocket * const watcher = mActiveSockets;
mActiveSockets = watcher->mActiveNext;
watcher->InvokeCallback();
}
}
void WatchableEventManager::Shutdown()
{
event_base_loopbreak(mEventBase);
event_free(mTimeoutEvent);
mTimeoutEvent = nullptr;
event_base_free(mEventBase);
mEventBase = nullptr;
}
// static
void WatchableEventManager::LibeventCallbackHandler(evutil_socket_t fd, short eventFlags, void * data)
{
WatchableSocket * const watcher = reinterpret_cast<WatchableSocket *>(data);
VerifyOrDie(watcher != nullptr);
VerifyOrDie(watcher->mFD == fd);
watcher->mPendingIO = SocketEventsFromLibeventFlags(eventFlags);
// Add to active list.
WatchableSocket ** pp = &watcher->mSharedState->mActiveSockets;
while (*pp != nullptr)
{
if (*pp == watcher)
{
return;
}
pp = &(*pp)->mActiveNext;
}
*pp = watcher;
watcher->mActiveNext = nullptr;
}
void WatchableEventManager::RemoveFromQueueIfPresent(WatchableSocket * watcher)
{
VerifyOrDie(watcher != nullptr);
VerifyOrDie(watcher->mSharedState == this);
WatchableSocket ** pp = &mActiveSockets;
while (*pp != nullptr)
{
if (*pp == watcher)
{
*pp = watcher->mActiveNext;
return;
}
pp = &(*pp)->mActiveNext;
}
}
void WatchableSocket::OnInit()
{
mEvent = nullptr;
mActiveNext = nullptr;
}
void WatchableSocket::OnAttach()
{
evutil_make_socket_nonblocking(mFD);
}
void WatchableSocket::SetWatch(short eventFlags)
{
const short oldFlags = mEvent ? event_get_events(mEvent) : 0;
const short newFlags = static_cast<short>(EV_PERSIST | oldFlags | eventFlags);
if (oldFlags != newFlags)
{
UpdateWatch(newFlags);
}
}
void WatchableSocket::ClearWatch(short eventFlags)
{
const short oldFlags = mEvent ? event_get_events(mEvent) : 0;
const short newFlags = static_cast<short>(EV_PERSIST | (oldFlags & ~eventFlags));
if (oldFlags != newFlags)
{
UpdateWatch(newFlags);
}
}
void WatchableSocket::UpdateWatch(short eventFlags)
{
if (mEvent)
{
event_del(mEvent);
event_free(mEvent);
mEvent = nullptr;
}
if (eventFlags)
{
event_base * const base = mSharedState->mEventBase;
mEvent = event_new(base, mFD, eventFlags, WatchableEventManager::LibeventCallbackHandler, this);
event_add(mEvent, nullptr);
}
}
} // namespace System
} // namespace chip