mirror of
https://gitlab.winehq.org/wine/wine-gecko.git
synced 2024-09-13 09:24:08 -07:00
e4e2da55c9
The bulk of this commit was generated with a script, executed at the top level of a typical source code checkout. The only non-machine-generated part was modifying MFBT's moz.build to reflect the new naming. CLOSED TREE makes big refactorings like this a piece of cake. # The main substitution. find . -name '*.cpp' -o -name '*.cc' -o -name '*.h' -o -name '*.mm' -o -name '*.idl'| \ xargs perl -p -i -e ' s/nsRefPtr\.h/RefPtr\.h/g; # handle includes s/nsRefPtr ?</RefPtr</g; # handle declarations and variables ' # Handle a special friend declaration in gfx/layers/AtomicRefCountedWithFinalize.h. perl -p -i -e 's/::nsRefPtr;/::RefPtr;/' gfx/layers/AtomicRefCountedWithFinalize.h # Handle nsRefPtr.h itself, a couple places that define constructors # from nsRefPtr, and code generators specially. We do this here, rather # than indiscriminantly s/nsRefPtr/RefPtr/, because that would rename # things like nsRefPtrHashtable. perl -p -i -e 's/nsRefPtr/RefPtr/g' \ mfbt/nsRefPtr.h \ xpcom/glue/nsCOMPtr.h \ xpcom/base/OwningNonNull.h \ ipc/ipdl/ipdl/lower.py \ ipc/ipdl/ipdl/builtin.py \ dom/bindings/Codegen.py \ python/lldbutils/lldbutils/utils.py # In our indiscriminate substitution above, we renamed # nsRefPtrGetterAddRefs, the class behind getter_AddRefs. Fix that up. find . -name '*.cpp' -o -name '*.h' -o -name '*.idl' | \ xargs perl -p -i -e 's/nsRefPtrGetterAddRefs/RefPtrGetterAddRefs/g' if [ -d .git ]; then git mv mfbt/nsRefPtr.h mfbt/RefPtr.h else hg mv mfbt/nsRefPtr.h mfbt/RefPtr.h fi
171 lines
4.7 KiB
C++
171 lines
4.7 KiB
C++
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/* vim:set ts=2 sw=2 sts=2 et cindent: */
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
#if !defined(MediaTimer_h_)
|
|
#define MediaTimer_h_
|
|
|
|
#include "mozilla/Monitor.h"
|
|
#include "mozilla/MozPromise.h"
|
|
#include "mozilla/TimeStamp.h"
|
|
|
|
#include <queue>
|
|
|
|
#include "nsITimer.h"
|
|
#include "mozilla/RefPtr.h"
|
|
|
|
namespace mozilla {
|
|
|
|
extern PRLogModuleInfo* gMediaTimerLog;
|
|
|
|
#define TIMER_LOG(x, ...) \
|
|
MOZ_ASSERT(gMediaTimerLog); \
|
|
MOZ_LOG(gMediaTimerLog, LogLevel::Debug, ("[MediaTimer=%p relative_t=%lld]" x, this, \
|
|
RelativeMicroseconds(TimeStamp::Now()), ##__VA_ARGS__))
|
|
|
|
// This promise type is only exclusive because so far there isn't a reason for
|
|
// it not to be. Feel free to change that.
|
|
typedef MozPromise<bool, bool, /* IsExclusive = */ true> MediaTimerPromise;
|
|
|
|
// Timers only know how to fire at a given thread, which creates an impedence
|
|
// mismatch with code that operates with TaskQueues. This class solves
|
|
// that mismatch with a dedicated (but shared) thread and a nice MozPromise-y
|
|
// interface.
|
|
class MediaTimer
|
|
{
|
|
public:
|
|
MediaTimer();
|
|
|
|
// We use a release with a custom Destroy().
|
|
NS_IMETHOD_(MozExternalRefCountType) AddRef(void);
|
|
NS_IMETHOD_(MozExternalRefCountType) Release(void);
|
|
|
|
RefPtr<MediaTimerPromise> WaitUntil(const TimeStamp& aTimeStamp, const char* aCallSite);
|
|
|
|
private:
|
|
virtual ~MediaTimer() { MOZ_ASSERT(OnMediaTimerThread()); }
|
|
|
|
void DispatchDestroy(); // Invoked by Release on an arbitrary thread.
|
|
void Destroy(); // Runs on the timer thread.
|
|
|
|
bool OnMediaTimerThread();
|
|
void ScheduleUpdate();
|
|
void Update();
|
|
void UpdateLocked();
|
|
|
|
static void TimerCallback(nsITimer* aTimer, void* aClosure);
|
|
void TimerFired();
|
|
void ArmTimer(const TimeStamp& aTarget, const TimeStamp& aNow);
|
|
|
|
bool TimerIsArmed()
|
|
{
|
|
return !mCurrentTimerTarget.IsNull();
|
|
}
|
|
|
|
void CancelTimerIfArmed()
|
|
{
|
|
MOZ_ASSERT(OnMediaTimerThread());
|
|
if (TimerIsArmed()) {
|
|
TIMER_LOG("MediaTimer::CancelTimerIfArmed canceling timer");
|
|
mTimer->Cancel();
|
|
mCurrentTimerTarget = TimeStamp();
|
|
}
|
|
}
|
|
|
|
|
|
struct Entry
|
|
{
|
|
TimeStamp mTimeStamp;
|
|
RefPtr<MediaTimerPromise::Private> mPromise;
|
|
|
|
explicit Entry(const TimeStamp& aTimeStamp, const char* aCallSite)
|
|
: mTimeStamp(aTimeStamp)
|
|
, mPromise(new MediaTimerPromise::Private(aCallSite))
|
|
{}
|
|
|
|
// Define a < overload that reverses ordering because std::priority_queue
|
|
// provides access to the largest element, and we want the smallest
|
|
// (i.e. the soonest).
|
|
bool operator<(const Entry& aOther) const
|
|
{
|
|
return mTimeStamp > aOther.mTimeStamp;
|
|
}
|
|
};
|
|
|
|
ThreadSafeAutoRefCnt mRefCnt;
|
|
NS_DECL_OWNINGTHREAD
|
|
nsCOMPtr<nsIEventTarget> mThread;
|
|
std::priority_queue<Entry> mEntries;
|
|
Monitor mMonitor;
|
|
nsCOMPtr<nsITimer> mTimer;
|
|
TimeStamp mCurrentTimerTarget;
|
|
|
|
// Timestamps only have relative meaning, so we need a base timestamp for
|
|
// logging purposes.
|
|
TimeStamp mCreationTimeStamp;
|
|
int64_t RelativeMicroseconds(const TimeStamp& aTimeStamp)
|
|
{
|
|
return (int64_t) (aTimeStamp - mCreationTimeStamp).ToMicroseconds();
|
|
}
|
|
|
|
bool mUpdateScheduled;
|
|
};
|
|
|
|
// Class for managing delayed dispatches on target thread.
|
|
class DelayedScheduler {
|
|
public:
|
|
explicit DelayedScheduler(AbstractThread* aTargetThread)
|
|
: mTargetThread(aTargetThread), mMediaTimer(new MediaTimer())
|
|
{
|
|
MOZ_ASSERT(mTargetThread);
|
|
}
|
|
|
|
bool IsScheduled() const { return !mTarget.IsNull(); }
|
|
|
|
void Reset()
|
|
{
|
|
MOZ_ASSERT(mTargetThread->IsCurrentThreadIn(),
|
|
"Must be on target thread to disconnect");
|
|
if (IsScheduled()) {
|
|
mRequest.Disconnect();
|
|
mTarget = TimeStamp();
|
|
}
|
|
}
|
|
|
|
template <typename ResolveFunc, typename RejectFunc>
|
|
void Ensure(mozilla::TimeStamp& aTarget,
|
|
ResolveFunc&& aResolver,
|
|
RejectFunc&& aRejector)
|
|
{
|
|
MOZ_ASSERT(mTargetThread->IsCurrentThreadIn());
|
|
if (IsScheduled() && mTarget <= aTarget) {
|
|
return;
|
|
}
|
|
Reset();
|
|
mTarget = aTarget;
|
|
mRequest.Begin(mMediaTimer->WaitUntil(mTarget, __func__)->Then(
|
|
mTargetThread, __func__,
|
|
Forward<ResolveFunc>(aResolver),
|
|
Forward<RejectFunc>(aRejector)));
|
|
}
|
|
|
|
void CompleteRequest()
|
|
{
|
|
MOZ_ASSERT(mTargetThread->IsCurrentThreadIn());
|
|
mRequest.Complete();
|
|
mTarget = TimeStamp();
|
|
}
|
|
|
|
private:
|
|
RefPtr<AbstractThread> mTargetThread;
|
|
RefPtr<MediaTimer> mMediaTimer;
|
|
MozPromiseRequestHolder<mozilla::MediaTimerPromise> mRequest;
|
|
TimeStamp mTarget;
|
|
};
|
|
|
|
} // namespace mozilla
|
|
|
|
#endif
|