blob: a9a3b3adca172941cb418aeb95cb9f16016e0242 [file]
//
// Copyright 2018 The ANGLE Project Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
//
// Observer:
// Implements the Observer pattern for sending state change notifications
// from Subject objects to dependent Observer objects.
//
// See design document:
// https://docs.google.com/document/d/15Edfotqg6_l1skTEL8ADQudF_oIdNa7i8Po43k6jMd4/
#include "libANGLE/Observer.h"
#include <algorithm>
#include "common/debug.h"
namespace angle
{
namespace
{
template <typename HaystackT, typename NeedleT>
bool IsInContainer(const HaystackT &haystack, const NeedleT &needle)
{
return std::find(haystack.begin(), haystack.end(), needle) != haystack.end();
}
} // anonymous namespace
// Observer implementation.
ObserverInterface::~ObserverInterface() = default;
// Subject implementation.
Subject::Subject()
{
}
Subject::~Subject()
{
resetObservers();
}
bool Subject::hasObservers() const
{
return !mFastObservers.empty();
}
void Subject::addObserver(ObserverBinding *observer)
{
ASSERT(!IsInContainer(mFastObservers, observer) && !IsInContainer(mSlowObservers, observer));
if (!mFastObservers.full())
{
mFastObservers.push_back(observer);
}
else
{
mSlowObservers.push_back(observer);
}
}
void Subject::removeObserver(ObserverBinding *observer)
{
auto iter = std::find(mFastObservers.begin(), mFastObservers.end(), observer);
if (iter != mFastObservers.end())
{
size_t index = iter - mFastObservers.begin();
std::swap(mFastObservers[index], mFastObservers[mFastObservers.size() - 1]);
mFastObservers.resize(mFastObservers.size() - 1);
if (!mSlowObservers.empty())
{
mFastObservers.push_back(mSlowObservers.back());
mSlowObservers.pop_back();
ASSERT(mFastObservers.full());
}
}
else
{
auto slowIter = std::find(mSlowObservers.begin(), mSlowObservers.end(), observer);
ASSERT(slowIter != mSlowObservers.end());
mSlowObservers.erase(slowIter);
}
}
void Subject::onStateChange(const gl::Context *context, SubjectMessage message) const
{
if (mFastObservers.empty())
return;
for (const angle::ObserverBinding *receiver : mFastObservers)
{
receiver->onStateChange(context, message);
}
for (const angle::ObserverBinding *receiver : mSlowObservers)
{
receiver->onStateChange(context, message);
}
}
void Subject::resetObservers()
{
for (angle::ObserverBinding *observer : mFastObservers)
{
observer->onSubjectReset();
}
mFastObservers.clear();
for (angle::ObserverBinding *observer : mSlowObservers)
{
observer->onSubjectReset();
}
mSlowObservers.clear();
}
// ObserverBinding implementation.
ObserverBinding::ObserverBinding(ObserverInterface *observer, SubjectIndex index)
: mSubject(nullptr), mObserver(observer), mIndex(index)
{
ASSERT(observer);
}
ObserverBinding::~ObserverBinding()
{
reset();
}
ObserverBinding::ObserverBinding(const ObserverBinding &other) = default;
ObserverBinding &ObserverBinding::operator=(const ObserverBinding &other) = default;
void ObserverBinding::bind(Subject *subject)
{
ASSERT(mObserver);
if (mSubject)
{
mSubject->removeObserver(this);
}
mSubject = subject;
if (mSubject)
{
mSubject->addObserver(this);
}
}
void ObserverBinding::reset()
{
bind(nullptr);
}
void ObserverBinding::onStateChange(const gl::Context *context, SubjectMessage message) const
{
mObserver->onSubjectStateChange(context, mIndex, message);
}
void ObserverBinding::onSubjectReset()
{
mSubject = nullptr;
}
const Subject *ObserverBinding::getSubject() const
{
return mSubject;
}
} // namespace angle