mirror of
https://bitbucket.org/chromiumembedded/cef
synced 2025-01-20 21:44:06 +01:00
398 lines
15 KiB
C++
398 lines
15 KiB
C++
// Copyright (c) 2014 Marshall A. Greenblatt. Portions copyright (c) 2013
|
|
// Google Inc. All rights reserved.
|
|
//
|
|
// Redistribution and use in source and binary forms, with or without
|
|
// modification, are permitted provided that the following conditions are
|
|
// met:
|
|
//
|
|
// * Redistributions of source code must retain the above copyright
|
|
// notice, this list of conditions and the following disclaimer.
|
|
// * Redistributions in binary form must reproduce the above
|
|
// copyright notice, this list of conditions and the following disclaimer
|
|
// in the documentation and/or other materials provided with the
|
|
// distribution.
|
|
// * Neither the name of Google Inc. nor the name Chromium Embedded
|
|
// Framework nor the names of its contributors may be used to endorse
|
|
// or promote products derived from this software without specific prior
|
|
// written permission.
|
|
//
|
|
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
|
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
|
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
|
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
///
|
|
/// \file
|
|
/// A container for a list of callbacks. Provides callers the ability to
|
|
/// manually or automatically unregister callbacks at any time, including during
|
|
/// callback notification.
|
|
///
|
|
/// TYPICAL USAGE:
|
|
///
|
|
/// <pre>
|
|
/// class MyWidget {
|
|
/// public:
|
|
/// using CallbackList = base::RepeatingCallbackList<void(const Foo&)>;
|
|
///
|
|
/// // Registers |cb| to be called whenever NotifyFoo() is executed.
|
|
/// CallbackListSubscription RegisterCallback(CallbackList::CallbackType cb) {
|
|
/// return callback_list_.Add(std::move(cb));
|
|
/// }
|
|
///
|
|
/// private:
|
|
/// // Calls all registered callbacks, with |foo| as the supplied arg.
|
|
/// void NotifyFoo(const Foo& foo) {
|
|
/// callback_list_.Notify(foo);
|
|
/// }
|
|
///
|
|
/// CallbackList callback_list_;
|
|
/// };
|
|
///
|
|
///
|
|
/// class MyWidgetListener {
|
|
/// private:
|
|
/// void OnFoo(const Foo& foo) {
|
|
/// // Called whenever MyWidget::NotifyFoo() is executed, unless
|
|
/// // |foo_subscription_| has been destroyed.
|
|
/// }
|
|
///
|
|
/// // Automatically deregisters the callback when deleted (e.g. in
|
|
/// // ~MyWidgetListener()). Unretained(this) is safe here since the
|
|
/// // ScopedClosureRunner does not outlive |this|.
|
|
/// CallbackListSubscription foo_subscription_ =
|
|
/// MyWidget::Get()->RegisterCallback(
|
|
/// base::BindRepeating(&MyWidgetListener::OnFoo,
|
|
/// base::Unretained(this)));
|
|
/// };
|
|
/// </pre>
|
|
///
|
|
/// UNSUPPORTED:
|
|
///
|
|
/// * Destroying the CallbackList during callback notification.
|
|
///
|
|
/// This is possible to support, but not currently necessary.
|
|
///
|
|
|
|
#ifndef CEF_INCLUDE_BASE_CEF_CALLBACK_LIST_H_
|
|
#define CEF_INCLUDE_BASE_CEF_CALLBACK_LIST_H_
|
|
#pragma once
|
|
|
|
#if defined(USING_CHROMIUM_INCLUDES)
|
|
// When building CEF include the Chromium header directly.
|
|
#include "base/callback_list.h"
|
|
#else // !USING_CHROMIUM_INCLUDES
|
|
// The following is substantially similar to the Chromium implementation.
|
|
// If the Chromium implementation diverges the below implementation should be
|
|
// updated to match.
|
|
|
|
#include <algorithm>
|
|
#include <list>
|
|
#include <memory>
|
|
#include <utility>
|
|
|
|
#include "include/base/cef_auto_reset.h"
|
|
#include "include/base/cef_bind.h"
|
|
#include "include/base/cef_callback.h"
|
|
#include "include/base/cef_callback_helpers.h"
|
|
#include "include/base/cef_logging.h"
|
|
#include "include/base/cef_weak_ptr.h"
|
|
|
|
namespace base {
|
|
namespace internal {
|
|
template <typename CallbackListImpl>
|
|
class CallbackListBase;
|
|
} // namespace internal
|
|
|
|
template <typename Signature>
|
|
class OnceCallbackList;
|
|
|
|
template <typename Signature>
|
|
class RepeatingCallbackList;
|
|
|
|
// A trimmed-down version of ScopedClosureRunner that can be used to guarantee a
|
|
// closure is run on destruction. This is designed to be used by
|
|
// CallbackListBase to run CancelCallback() when this subscription dies;
|
|
// consumers can avoid callbacks on dead objects by ensuring the subscription
|
|
// returned by CallbackListBase::Add() does not outlive the bound object in the
|
|
// callback. A typical way to do this is to bind a callback to a member function
|
|
// on `this` and store the returned subscription as a member variable.
|
|
class CallbackListSubscription {
|
|
public:
|
|
CallbackListSubscription();
|
|
CallbackListSubscription(CallbackListSubscription&& subscription);
|
|
CallbackListSubscription& operator=(CallbackListSubscription&& subscription);
|
|
~CallbackListSubscription();
|
|
|
|
explicit operator bool() const { return !!closure_; }
|
|
|
|
private:
|
|
template <typename T>
|
|
friend class internal::CallbackListBase;
|
|
|
|
explicit CallbackListSubscription(base::OnceClosure closure);
|
|
|
|
void Run();
|
|
|
|
OnceClosure closure_;
|
|
};
|
|
|
|
namespace internal {
|
|
|
|
// From base/stl_util.h.
|
|
template <class T, class Allocator, class Predicate>
|
|
size_t EraseIf(std::list<T, Allocator>& container, Predicate pred) {
|
|
size_t old_size = container.size();
|
|
container.remove_if(pred);
|
|
return old_size - container.size();
|
|
}
|
|
|
|
// A traits class to break circular type dependencies between CallbackListBase
|
|
// and its subclasses.
|
|
template <typename CallbackList>
|
|
struct CallbackListTraits;
|
|
|
|
// NOTE: It's important that Callbacks provide iterator stability when items are
|
|
// added to the end, so e.g. a std::vector<> is not suitable here.
|
|
template <typename Signature>
|
|
struct CallbackListTraits<OnceCallbackList<Signature>> {
|
|
using CallbackType = OnceCallback<Signature>;
|
|
using Callbacks = std::list<CallbackType>;
|
|
};
|
|
template <typename Signature>
|
|
struct CallbackListTraits<RepeatingCallbackList<Signature>> {
|
|
using CallbackType = RepeatingCallback<Signature>;
|
|
using Callbacks = std::list<CallbackType>;
|
|
};
|
|
|
|
template <typename CallbackListImpl>
|
|
class CallbackListBase {
|
|
public:
|
|
using CallbackType =
|
|
typename CallbackListTraits<CallbackListImpl>::CallbackType;
|
|
static_assert(IsBaseCallback<CallbackType>::value, "");
|
|
|
|
// TODO(crbug.com/1103086): Update references to use this directly and by
|
|
// value, then remove.
|
|
using Subscription = CallbackListSubscription;
|
|
|
|
CallbackListBase() = default;
|
|
CallbackListBase(const CallbackListBase&) = delete;
|
|
CallbackListBase& operator=(const CallbackListBase&) = delete;
|
|
|
|
~CallbackListBase() {
|
|
// Destroying the list during iteration is unsupported and will cause a UAF.
|
|
CHECK(!iterating_);
|
|
}
|
|
|
|
// Registers |cb| for future notifications. Returns a CallbackListSubscription
|
|
// whose destruction will cancel |cb|.
|
|
[[nodiscard]] CallbackListSubscription Add(CallbackType cb) {
|
|
DCHECK(!cb.is_null());
|
|
return CallbackListSubscription(base::BindOnce(
|
|
&CallbackListBase::CancelCallback, weak_ptr_factory_.GetWeakPtr(),
|
|
callbacks_.insert(callbacks_.end(), std::move(cb))));
|
|
}
|
|
|
|
// Registers |cb| for future notifications. Provides no way for the caller to
|
|
// cancel, so this is only safe for cases where the callback is guaranteed to
|
|
// live at least as long as this list (e.g. if it's bound on the same object
|
|
// that owns the list).
|
|
// TODO(pkasting): Attempt to use Add() instead and see if callers can relax
|
|
// other lifetime/ordering mechanisms as a result.
|
|
void AddUnsafe(CallbackType cb) {
|
|
DCHECK(!cb.is_null());
|
|
callbacks_.push_back(std::move(cb));
|
|
}
|
|
|
|
// Registers |removal_callback| to be run after elements are removed from the
|
|
// list of registered callbacks.
|
|
void set_removal_callback(const RepeatingClosure& removal_callback) {
|
|
removal_callback_ = removal_callback;
|
|
}
|
|
|
|
// Returns whether the list of registered callbacks is empty (from an external
|
|
// perspective -- meaning no remaining callbacks are live).
|
|
bool empty() const {
|
|
return std::all_of(callbacks_.cbegin(), callbacks_.cend(),
|
|
[](const auto& callback) { return callback.is_null(); });
|
|
}
|
|
|
|
// Calls all registered callbacks that are not canceled beforehand. If any
|
|
// callbacks are unregistered, notifies any registered removal callback at the
|
|
// end.
|
|
//
|
|
// Arguments must be copyable, since they must be supplied to all callbacks.
|
|
// Move-only types would be destructively modified by passing them to the
|
|
// first callback and not reach subsequent callbacks as intended.
|
|
//
|
|
// Notify() may be called re-entrantly, in which case the nested call
|
|
// completes before the outer one continues. Callbacks are only ever added at
|
|
// the end and canceled callbacks are not pruned from the list until the
|
|
// outermost iteration completes, so existing iterators should never be
|
|
// invalidated. However, this does mean that a callback added during a nested
|
|
// call can be notified by outer calls -- meaning it will be notified about
|
|
// things that happened before it was added -- if its subscription outlives
|
|
// the reentrant Notify() call.
|
|
template <typename... RunArgs>
|
|
void Notify(RunArgs&&... args) {
|
|
if (empty())
|
|
return; // Nothing to do.
|
|
|
|
{
|
|
AutoReset<bool> iterating(&iterating_, true);
|
|
|
|
// Skip any callbacks that are canceled during iteration.
|
|
// NOTE: Since RunCallback() may call Add(), it's not safe to cache the
|
|
// value of callbacks_.end() across loop iterations.
|
|
const auto next_valid = [this](const auto it) {
|
|
return std::find_if_not(it, callbacks_.end(), [](const auto& callback) {
|
|
return callback.is_null();
|
|
});
|
|
};
|
|
for (auto it = next_valid(callbacks_.begin()); it != callbacks_.end();
|
|
it = next_valid(it))
|
|
// NOTE: Intentionally does not call std::forward<RunArgs>(args)...,
|
|
// since that would allow move-only arguments.
|
|
static_cast<CallbackListImpl*>(this)->RunCallback(it++, args...);
|
|
}
|
|
|
|
// Re-entrant invocations shouldn't prune anything from the list. This can
|
|
// invalidate iterators from underneath higher call frames. It's safe to
|
|
// simply do nothing, since the outermost frame will continue through here
|
|
// and prune all null callbacks below.
|
|
if (iterating_)
|
|
return;
|
|
|
|
// Any null callbacks remaining in the list were canceled due to
|
|
// Subscription destruction during iteration, and can safely be erased now.
|
|
const size_t erased_callbacks =
|
|
EraseIf(callbacks_, [](const auto& cb) { return cb.is_null(); });
|
|
|
|
// Run |removal_callback_| if any callbacks were canceled. Note that we
|
|
// cannot simply compare list sizes before and after iterating, since
|
|
// notification may result in Add()ing new callbacks as well as canceling
|
|
// them. Also note that if this is a OnceCallbackList, the OnceCallbacks
|
|
// that were executed above have all been removed regardless of whether
|
|
// they're counted in |erased_callbacks_|.
|
|
if (removal_callback_ &&
|
|
(erased_callbacks || IsOnceCallback<CallbackType>::value))
|
|
removal_callback_.Run(); // May delete |this|!
|
|
}
|
|
|
|
protected:
|
|
using Callbacks = typename CallbackListTraits<CallbackListImpl>::Callbacks;
|
|
|
|
// Holds non-null callbacks, which will be called during Notify().
|
|
Callbacks callbacks_;
|
|
|
|
private:
|
|
// Cancels the callback pointed to by |it|, which is guaranteed to be valid.
|
|
void CancelCallback(const typename Callbacks::iterator& it) {
|
|
if (static_cast<CallbackListImpl*>(this)->CancelNullCallback(it))
|
|
return;
|
|
|
|
if (iterating_) {
|
|
// Calling erase() here is unsafe, since the loop in Notify() may be
|
|
// referencing this same iterator, e.g. if adjacent callbacks'
|
|
// Subscriptions are both destroyed when the first one is Run(). Just
|
|
// reset the callback and let Notify() clean it up at the end.
|
|
it->Reset();
|
|
} else {
|
|
callbacks_.erase(it);
|
|
if (removal_callback_)
|
|
removal_callback_.Run(); // May delete |this|!
|
|
}
|
|
}
|
|
|
|
// Set while Notify() is traversing |callbacks_|. Used primarily to avoid
|
|
// invalidating iterators that may be in use.
|
|
bool iterating_ = false;
|
|
|
|
// Called after elements are removed from |callbacks_|.
|
|
RepeatingClosure removal_callback_;
|
|
|
|
WeakPtrFactory<CallbackListBase> weak_ptr_factory_{this};
|
|
};
|
|
|
|
} // namespace internal
|
|
|
|
template <typename Signature>
|
|
class OnceCallbackList
|
|
: public internal::CallbackListBase<OnceCallbackList<Signature>> {
|
|
private:
|
|
friend internal::CallbackListBase<OnceCallbackList>;
|
|
using Traits = internal::CallbackListTraits<OnceCallbackList>;
|
|
|
|
// Runs the current callback, which may cancel it or any other callbacks.
|
|
template <typename... RunArgs>
|
|
void RunCallback(typename Traits::Callbacks::iterator it, RunArgs&&... args) {
|
|
// OnceCallbacks still have Subscriptions with outstanding iterators;
|
|
// splice() removes them from |callbacks_| without invalidating those.
|
|
null_callbacks_.splice(null_callbacks_.end(), this->callbacks_, it);
|
|
|
|
// NOTE: Intentionally does not call std::forward<RunArgs>(args)...; see
|
|
// comments in Notify().
|
|
std::move(*it).Run(args...);
|
|
}
|
|
|
|
// If |it| refers to an already-canceled callback, does any necessary cleanup
|
|
// and returns true. Otherwise returns false.
|
|
bool CancelNullCallback(const typename Traits::Callbacks::iterator& it) {
|
|
if (it->is_null()) {
|
|
null_callbacks_.erase(it);
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
// Holds null callbacks whose Subscriptions are still alive, so the
|
|
// Subscriptions will still contain valid iterators. Only needed for
|
|
// OnceCallbacks, since RepeatingCallbacks are not canceled except by
|
|
// Subscription destruction.
|
|
typename Traits::Callbacks null_callbacks_;
|
|
};
|
|
|
|
template <typename Signature>
|
|
class RepeatingCallbackList
|
|
: public internal::CallbackListBase<RepeatingCallbackList<Signature>> {
|
|
private:
|
|
friend internal::CallbackListBase<RepeatingCallbackList>;
|
|
using Traits = internal::CallbackListTraits<RepeatingCallbackList>;
|
|
// Runs the current callback, which may cancel it or any other callbacks.
|
|
template <typename... RunArgs>
|
|
void RunCallback(typename Traits::Callbacks::iterator it, RunArgs&&... args) {
|
|
// NOTE: Intentionally does not call std::forward<RunArgs>(args)...; see
|
|
// comments in Notify().
|
|
it->Run(args...);
|
|
}
|
|
|
|
// If |it| refers to an already-canceled callback, does any necessary cleanup
|
|
// and returns true. Otherwise returns false.
|
|
bool CancelNullCallback(const typename Traits::Callbacks::iterator& it) {
|
|
// Because at most one Subscription can point to a given callback, and
|
|
// RepeatingCallbacks are only reset by CancelCallback(), no one should be
|
|
// able to request cancellation of a canceled RepeatingCallback.
|
|
DCHECK(!it->is_null());
|
|
return false;
|
|
}
|
|
};
|
|
|
|
///
|
|
/// Syntactic sugar to parallel that used for Callbacks.
|
|
///
|
|
using OnceClosureList = OnceCallbackList<void()>;
|
|
using RepeatingClosureList = RepeatingCallbackList<void()>;
|
|
|
|
} // namespace base
|
|
|
|
#endif // !USING_CHROMIUM_INCLUDES
|
|
|
|
#endif // CEF_INCLUDE_BASE_CEF_CALLBACK_LIST_H_
|