diff options
Diffstat (limited to '')
-rw-r--r-- | image/ProgressTracker.cpp | 556 |
1 files changed, 556 insertions, 0 deletions
diff --git a/image/ProgressTracker.cpp b/image/ProgressTracker.cpp new file mode 100644 index 0000000000..9338b8280b --- /dev/null +++ b/image/ProgressTracker.cpp @@ -0,0 +1,556 @@ +/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- + * + * 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/. */ + +#include "ImageLogging.h" +#include "ProgressTracker.h" + +#include "imgINotificationObserver.h" +#include "imgIRequest.h" +#include "Image.h" +#include "nsNetUtil.h" +#include "nsIObserverService.h" + +#include "mozilla/Assertions.h" +#include "mozilla/Services.h" + +using mozilla::WeakPtr; + +namespace mozilla { +namespace image { + +static void CheckProgressConsistency(Progress aOldProgress, + Progress aNewProgress, bool aIsMultipart) { + // Check preconditions for every progress bit. + + // Error's do not get propagated from the tracker for each image part to the + // tracker for the multipart image because we don't want one bad part to + // prevent the remaining parts from showing. So we need to consider whether + // this is a tracker for a multipart image for these assertions to work. + + if (aNewProgress & FLAG_SIZE_AVAILABLE) { + // No preconditions. + } + if (aNewProgress & FLAG_DECODE_COMPLETE) { + MOZ_ASSERT(aNewProgress & FLAG_SIZE_AVAILABLE); + MOZ_ASSERT(aIsMultipart || + aNewProgress & (FLAG_FRAME_COMPLETE | FLAG_HAS_ERROR)); + } + if (aNewProgress & FLAG_FRAME_COMPLETE) { + MOZ_ASSERT(aNewProgress & FLAG_SIZE_AVAILABLE); + } + if (aNewProgress & FLAG_LOAD_COMPLETE) { + MOZ_ASSERT(aIsMultipart || + aNewProgress & (FLAG_SIZE_AVAILABLE | FLAG_HAS_ERROR)); + } + if (aNewProgress & FLAG_IS_ANIMATED) { + // No preconditions; like FLAG_HAS_TRANSPARENCY, we should normally never + // discover this *after* FLAG_SIZE_AVAILABLE, but unfortunately some corrupt + // GIFs may fool us. + } + if (aNewProgress & FLAG_HAS_TRANSPARENCY) { + // XXX We'd like to assert that transparency is only set during metadata + // decode but we don't have any way to assert that until bug 1254892 is + // fixed. + } + if (aNewProgress & FLAG_LAST_PART_COMPLETE) { + MOZ_ASSERT(aNewProgress & FLAG_LOAD_COMPLETE); + } + if (aNewProgress & FLAG_HAS_ERROR) { + // No preconditions. + } +} + +ProgressTracker::ProgressTracker() + : mMutex("ProgressTracker::mMutex"), + mImage(nullptr), + mEventTarget(WrapNotNull( + nsCOMPtr<nsIEventTarget>(GetMainThreadSerialEventTarget()))), + mObserversWithTargets(0), + mObservers(new ObserverTable), + mProgress(NoProgress), + mIsMultipart(false) {} + +void ProgressTracker::SetImage(Image* aImage) { + MutexAutoLock lock(mMutex); + MOZ_ASSERT(aImage, "Setting null image"); + MOZ_ASSERT(!mImage, "Setting image when we already have one"); + mImage = aImage; +} + +void ProgressTracker::ResetImage() { + MutexAutoLock lock(mMutex); + MOZ_ASSERT(mImage, "Resetting image when it's already null!"); + mImage = nullptr; +} + +uint32_t ProgressTracker::GetImageStatus() const { + uint32_t status = imgIRequest::STATUS_NONE; + + // Translate our current state to a set of imgIRequest::STATE_* flags. + if (mProgress & FLAG_SIZE_AVAILABLE) { + status |= imgIRequest::STATUS_SIZE_AVAILABLE; + } + if (mProgress & FLAG_DECODE_COMPLETE) { + status |= imgIRequest::STATUS_DECODE_COMPLETE; + } + if (mProgress & FLAG_FRAME_COMPLETE) { + status |= imgIRequest::STATUS_FRAME_COMPLETE; + } + if (mProgress & FLAG_LOAD_COMPLETE) { + status |= imgIRequest::STATUS_LOAD_COMPLETE; + } + if (mProgress & FLAG_IS_ANIMATED) { + status |= imgIRequest::STATUS_IS_ANIMATED; + } + if (mProgress & FLAG_HAS_TRANSPARENCY) { + status |= imgIRequest::STATUS_HAS_TRANSPARENCY; + } + if (mProgress & FLAG_HAS_ERROR) { + status |= imgIRequest::STATUS_ERROR; + } + + return status; +} + +// A helper class to allow us to call SyncNotify asynchronously. +class AsyncNotifyRunnable : public Runnable { + public: + AsyncNotifyRunnable(ProgressTracker* aTracker, IProgressObserver* aObserver) + : Runnable("ProgressTracker::AsyncNotifyRunnable"), mTracker(aTracker) { + MOZ_ASSERT(NS_IsMainThread(), "Should be created on the main thread"); + MOZ_ASSERT(aTracker, "aTracker should not be null"); + MOZ_ASSERT(aObserver, "aObserver should not be null"); + mObservers.AppendElement(aObserver); + } + + NS_IMETHOD Run() override { + MOZ_ASSERT(NS_IsMainThread(), "Should be running on the main thread"); + MOZ_ASSERT(mTracker, "mTracker should not be null"); + for (uint32_t i = 0; i < mObservers.Length(); ++i) { + mObservers[i]->ClearPendingNotify(); + mTracker->SyncNotify(mObservers[i]); + } + + mTracker->mRunnable = nullptr; + return NS_OK; + } + + void AddObserver(IProgressObserver* aObserver) { + mObservers.AppendElement(aObserver); + } + + void RemoveObserver(IProgressObserver* aObserver) { + mObservers.RemoveElement(aObserver); + } + + private: + friend class ProgressTracker; + + RefPtr<ProgressTracker> mTracker; + nsTArray<RefPtr<IProgressObserver>> mObservers; +}; + +ProgressTracker::MediumHighRunnable::MediumHighRunnable( + already_AddRefed<AsyncNotifyRunnable>&& aEvent) + : PrioritizableRunnable(std::move(aEvent), + nsIRunnablePriority::PRIORITY_MEDIUMHIGH) {} + +void ProgressTracker::MediumHighRunnable::AddObserver( + IProgressObserver* aObserver) { + static_cast<AsyncNotifyRunnable*>(mRunnable.get())->AddObserver(aObserver); +} + +void ProgressTracker::MediumHighRunnable::RemoveObserver( + IProgressObserver* aObserver) { + static_cast<AsyncNotifyRunnable*>(mRunnable.get())->RemoveObserver(aObserver); +} + +/* static */ +already_AddRefed<ProgressTracker::MediumHighRunnable> +ProgressTracker::MediumHighRunnable::Create( + already_AddRefed<AsyncNotifyRunnable>&& aEvent) { + MOZ_ASSERT(NS_IsMainThread()); + RefPtr<ProgressTracker::MediumHighRunnable> event( + new ProgressTracker::MediumHighRunnable(std::move(aEvent))); + return event.forget(); +} + +void ProgressTracker::Notify(IProgressObserver* aObserver) { + MOZ_ASSERT(NS_IsMainThread()); + + if (aObserver->NotificationsDeferred()) { + // There is a pending notification, or the observer isn't ready yet. + return; + } + + if (MOZ_LOG_TEST(gImgLog, LogLevel::Debug)) { + RefPtr<Image> image = GetImage(); + LOG_FUNC_WITH_PARAM(gImgLog, "ProgressTracker::Notify async", "uri", image); + } + + aObserver->MarkPendingNotify(); + + // If we have an existing runnable that we can use, we just append this + // observer to its list of observers to be notified. This ensures we don't + // unnecessarily delay onload. + if (mRunnable) { + mRunnable->AddObserver(aObserver); + } else { + RefPtr<AsyncNotifyRunnable> ev = new AsyncNotifyRunnable(this, aObserver); + mRunnable = ProgressTracker::MediumHighRunnable::Create(ev.forget()); + mEventTarget->Dispatch(mRunnable, NS_DISPATCH_NORMAL); + } +} + +// A helper class to allow us to call SyncNotify asynchronously for a given, +// fixed, state. +class AsyncNotifyCurrentStateRunnable : public Runnable { + public: + AsyncNotifyCurrentStateRunnable(ProgressTracker* aProgressTracker, + IProgressObserver* aObserver) + : Runnable("image::AsyncNotifyCurrentStateRunnable"), + mProgressTracker(aProgressTracker), + mObserver(aObserver) { + MOZ_ASSERT(NS_IsMainThread(), "Should be created on the main thread"); + MOZ_ASSERT(mProgressTracker, "mProgressTracker should not be null"); + MOZ_ASSERT(mObserver, "mObserver should not be null"); + mImage = mProgressTracker->GetImage(); + } + + NS_IMETHOD Run() override { + MOZ_ASSERT(NS_IsMainThread(), "Should be running on the main thread"); + mObserver->ClearPendingNotify(); + + mProgressTracker->SyncNotify(mObserver); + return NS_OK; + } + + private: + RefPtr<ProgressTracker> mProgressTracker; + RefPtr<IProgressObserver> mObserver; + + // We have to hold on to a reference to the tracker's image, just in case + // it goes away while we're in the event queue. + RefPtr<Image> mImage; +}; + +void ProgressTracker::NotifyCurrentState(IProgressObserver* aObserver) { + MOZ_ASSERT(NS_IsMainThread()); + + if (aObserver->NotificationsDeferred()) { + // There is a pending notification, or the observer isn't ready yet. + return; + } + + if (MOZ_LOG_TEST(gImgLog, LogLevel::Debug)) { + RefPtr<Image> image = GetImage(); + LOG_FUNC_WITH_PARAM(gImgLog, "ProgressTracker::NotifyCurrentState", "uri", + image); + } + + aObserver->MarkPendingNotify(); + + nsCOMPtr<nsIRunnable> ev = + new AsyncNotifyCurrentStateRunnable(this, aObserver); + mEventTarget->Dispatch(ev.forget(), NS_DISPATCH_NORMAL); +} + +/** + * ImageObserverNotifier is a helper type that abstracts over the difference + * between sending notifications to all of the observers in an ObserverTable, + * and sending them to a single observer. This allows the same notification code + * to be used for both cases. + */ +template <typename T> +struct ImageObserverNotifier; + +template <> +struct MOZ_STACK_CLASS ImageObserverNotifier<const ObserverTable*> { + explicit ImageObserverNotifier(const ObserverTable* aObservers, + bool aIgnoreDeferral = false) + : mObservers(aObservers), mIgnoreDeferral(aIgnoreDeferral) {} + + template <typename Lambda> + void operator()(Lambda aFunc) { + for (auto iter = mObservers->ConstIter(); !iter.Done(); iter.Next()) { + RefPtr<IProgressObserver> observer = iter.Data().get(); + if (observer && (mIgnoreDeferral || !observer->NotificationsDeferred())) { + aFunc(observer); + } + } + } + + private: + const ObserverTable* mObservers; + const bool mIgnoreDeferral; +}; + +template <> +struct MOZ_STACK_CLASS ImageObserverNotifier<IProgressObserver*> { + explicit ImageObserverNotifier(IProgressObserver* aObserver) + : mObserver(aObserver) {} + + template <typename Lambda> + void operator()(Lambda aFunc) { + if (mObserver && !mObserver->NotificationsDeferred()) { + aFunc(mObserver); + } + } + + private: + IProgressObserver* mObserver; +}; + +template <typename T> +void SyncNotifyInternal(const T& aObservers, bool aHasImage, Progress aProgress, + const nsIntRect& aDirtyRect) { + MOZ_ASSERT(NS_IsMainThread()); + + typedef imgINotificationObserver I; + ImageObserverNotifier<T> notify(aObservers); + + if (aProgress & FLAG_SIZE_AVAILABLE) { + notify([](IProgressObserver* aObs) { aObs->Notify(I::SIZE_AVAILABLE); }); + } + + if (aHasImage) { + // OnFrameUpdate + // If there's any content in this frame at all (always true for + // vector images, true for raster images that have decoded at + // least one frame) then send OnFrameUpdate. + if (!aDirtyRect.IsEmpty()) { + notify([&](IProgressObserver* aObs) { + aObs->Notify(I::FRAME_UPDATE, &aDirtyRect); + }); + } + + if (aProgress & FLAG_FRAME_COMPLETE) { + notify([](IProgressObserver* aObs) { aObs->Notify(I::FRAME_COMPLETE); }); + } + + if (aProgress & FLAG_HAS_TRANSPARENCY) { + notify( + [](IProgressObserver* aObs) { aObs->Notify(I::HAS_TRANSPARENCY); }); + } + + if (aProgress & FLAG_IS_ANIMATED) { + notify([](IProgressObserver* aObs) { aObs->Notify(I::IS_ANIMATED); }); + } + } + + if (aProgress & FLAG_DECODE_COMPLETE) { + MOZ_ASSERT(aHasImage, "Stopped decoding without ever having an image?"); + notify([](IProgressObserver* aObs) { aObs->Notify(I::DECODE_COMPLETE); }); + } + + if (aProgress & FLAG_LOAD_COMPLETE) { + notify([=](IProgressObserver* aObs) { + aObs->OnLoadComplete(aProgress & FLAG_LAST_PART_COMPLETE); + }); + } +} + +void ProgressTracker::SyncNotifyProgress(Progress aProgress, + const nsIntRect& aInvalidRect + /* = nsIntRect() */) { + MOZ_ASSERT(NS_IsMainThread(), "Use mObservers on main thread only"); + + Progress progress = Difference(aProgress); + CheckProgressConsistency(mProgress, mProgress | progress, mIsMultipart); + + // Apply the changes. + mProgress |= progress; + + // Send notifications. + mObservers.Read([&](const ObserverTable* aTable) { + SyncNotifyInternal(aTable, HasImage(), progress, aInvalidRect); + }); + + if (progress & FLAG_HAS_ERROR) { + FireFailureNotification(); + } +} + +void ProgressTracker::SyncNotify(IProgressObserver* aObserver) { + MOZ_ASSERT(NS_IsMainThread()); + + RefPtr<Image> image = GetImage(); + LOG_SCOPE_WITH_PARAM(gImgLog, "ProgressTracker::SyncNotify", "uri", image); + + nsIntRect rect; + if (image) { + int32_t width, height; + if (NS_FAILED(image->GetWidth(&width)) || + NS_FAILED(image->GetHeight(&height))) { + // Either the image has no intrinsic size, or it has an error. + rect = GetMaxSizedIntRect(); + } else { + rect.SizeTo(width, height); + } + } + + SyncNotifyInternal(aObserver, !!image, mProgress, rect); +} + +void ProgressTracker::EmulateRequestFinished(IProgressObserver* aObserver) { + MOZ_ASSERT(NS_IsMainThread(), + "SyncNotifyState and mObservers are not threadsafe"); + RefPtr<IProgressObserver> kungFuDeathGrip(aObserver); + + if (!(mProgress & FLAG_LOAD_COMPLETE)) { + aObserver->OnLoadComplete(true); + } +} + +already_AddRefed<nsIEventTarget> ProgressTracker::GetEventTarget() const { + MutexAutoLock lock(mMutex); + nsCOMPtr<nsIEventTarget> target = mEventTarget; + return target.forget(); +} + +void ProgressTracker::AddObserver(IProgressObserver* aObserver) { + MOZ_ASSERT(NS_IsMainThread()); + RefPtr<IProgressObserver> observer = aObserver; + + nsCOMPtr<nsIEventTarget> target = observer->GetEventTarget(); + if (target) { + if (mObserversWithTargets == 0) { + // On the first observer with a target (i.e. listener), always accept its + // event target; this may be for a specific DocGroup, or it may be the + // unlabelled main thread target. + MutexAutoLock lock(mMutex); + mEventTarget = WrapNotNull(target); + } else if (mEventTarget.get() != target.get()) { + // If a subsequent observer comes in with a different target, we need to + // switch to use the unlabelled main thread target, if we haven't already. + MutexAutoLock lock(mMutex); + nsCOMPtr<nsIEventTarget> mainTarget(do_GetMainThread()); + mEventTarget = WrapNotNull(mainTarget); + } + ++mObserversWithTargets; + } + + mObservers.Write([=](ObserverTable* aTable) { + MOZ_ASSERT(!aTable->Get(observer, nullptr), + "Adding duplicate entry for image observer"); + + WeakPtr<IProgressObserver> weakPtr = observer.get(); + aTable->Put(observer, weakPtr); + }); + + MOZ_ASSERT(mObserversWithTargets <= ObserverCount()); +} + +bool ProgressTracker::RemoveObserver(IProgressObserver* aObserver) { + MOZ_ASSERT(NS_IsMainThread()); + RefPtr<IProgressObserver> observer = aObserver; + + // Remove the observer from the list. + bool removed = mObservers.Write( + [observer](ObserverTable* aTable) { return aTable->Remove(observer); }); + + // Sometimes once an image is decoded, and all of its observers removed, a new + // document may request the same image. Thus we need to clear our event target + // state when the last observer is removed, so that we select the most + // appropriate event target when a new observer is added. Since the event + // target may have changed (e.g. due to the scheduler group going away before + // we were removed), so we should be cautious comparing this target against + // anything at this stage. + if (removed) { + nsCOMPtr<nsIEventTarget> target = observer->GetEventTarget(); + if (target) { + MOZ_ASSERT(mObserversWithTargets > 0); + --mObserversWithTargets; + + // If we've shutdown the main thread there's no need to update + // event targets. + if ((mObserversWithTargets == 0) && !gXPCOMThreadsShutDown) { + MutexAutoLock lock(mMutex); + nsCOMPtr<nsIEventTarget> target(do_GetMainThread()); + mEventTarget = WrapNotNull(target); + } + } + + MOZ_ASSERT(mObserversWithTargets <= ObserverCount()); + } + + // Observers can get confused if they don't get all the proper teardown + // notifications. Part ways on good terms. + if (removed && !aObserver->NotificationsDeferred()) { + EmulateRequestFinished(aObserver); + } + + // Make sure we don't give callbacks to an observer that isn't interested in + // them any more. + if (aObserver->NotificationsDeferred() && mRunnable) { + mRunnable->RemoveObserver(aObserver); + aObserver->ClearPendingNotify(); + } + + return removed; +} + +uint32_t ProgressTracker::ObserverCount() const { + MOZ_ASSERT(NS_IsMainThread()); + return mObservers.Read( + [](const ObserverTable* aTable) { return aTable->Count(); }); +} + +void ProgressTracker::OnUnlockedDraw() { + MOZ_ASSERT(NS_IsMainThread()); + mObservers.Read([](const ObserverTable* aTable) { + ImageObserverNotifier<const ObserverTable*> notify(aTable); + notify([](IProgressObserver* aObs) { + aObs->Notify(imgINotificationObserver::UNLOCKED_DRAW); + }); + }); +} + +void ProgressTracker::ResetForNewRequest() { + MOZ_ASSERT(NS_IsMainThread()); + mProgress = NoProgress; +} + +void ProgressTracker::OnDiscard() { + MOZ_ASSERT(NS_IsMainThread()); + mObservers.Read([](const ObserverTable* aTable) { + ImageObserverNotifier<const ObserverTable*> notify(aTable); + notify([](IProgressObserver* aObs) { + aObs->Notify(imgINotificationObserver::DISCARD); + }); + }); +} + +void ProgressTracker::OnImageAvailable() { + MOZ_ASSERT(NS_IsMainThread()); + // Notify any imgRequestProxys that are observing us that we have an Image. + mObservers.Read([](const ObserverTable* aTable) { + ImageObserverNotifier<const ObserverTable*> notify( + aTable, /* aIgnoreDeferral = */ true); + notify([](IProgressObserver* aObs) { aObs->SetHasImage(); }); + }); +} + +void ProgressTracker::FireFailureNotification() { + MOZ_ASSERT(NS_IsMainThread()); + + // Some kind of problem has happened with image decoding. + // Report the URI to net:failed-to-process-uri-conent observers. + RefPtr<Image> image = GetImage(); + if (image) { + // Should be on main thread, so ok to create a new nsIURI. + nsCOMPtr<nsIURI> uri = image->GetURI(); + if (uri) { + nsCOMPtr<nsIObserverService> os = mozilla::services::GetObserverService(); + if (os) { + os->NotifyObservers(uri, "net:failed-to-process-uri-content", nullptr); + } + } + } +} + +} // namespace image +} // namespace mozilla |