forked from mirrors/gecko-dev
		
	 ebb8fa1e75
			
		
	
	
		ebb8fa1e75
		
	
	
	
	
		
			
			My guess is that if Observe() gets called too early, we can receive the message before updating the vsync source the first time, which would trigger this crash. With this patch, we handle it nicely and UpdateVsyncSource will re-observe as needed. It's not 100% clear to me if this is actually the issue (and thus whether this patch is the right solution), since I haven't been able to repro locally, though. There might be a more subtle race. Differential Revision: https://phabricator.services.mozilla.com/D100501
		
			
				
	
	
		
			107 lines
		
	
	
	
		
			3.2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			107 lines
		
	
	
	
		
			3.2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
 | |
| /* vim: set ts=8 sts=2 et sw=2 tw=80: */
 | |
| /* 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 "VsyncParent.h"
 | |
| 
 | |
| #include "BackgroundParent.h"
 | |
| #include "BackgroundParentImpl.h"
 | |
| #include "gfxPlatform.h"
 | |
| #include "mozilla/Unused.h"
 | |
| #include "nsThreadUtils.h"
 | |
| #include "VsyncSource.h"
 | |
| 
 | |
| namespace mozilla::dom {
 | |
| 
 | |
| VsyncParent::VsyncParent()
 | |
|     : mObservingVsync(false),
 | |
|       mDestroyed(false),
 | |
|       mInitialThread(NS_GetCurrentThread()) {}
 | |
| 
 | |
| void VsyncParent::UpdateVsyncSource(
 | |
|     const RefPtr<gfx::VsyncSource>& aVsyncSource) {
 | |
|   mVsyncSource = aVsyncSource;
 | |
|   if (!mVsyncSource) {
 | |
|     mVsyncSource = gfxPlatform::GetPlatform()->GetHardwareVsync();
 | |
|   }
 | |
| 
 | |
|   if (mObservingVsync && mVsyncDispatcher) {
 | |
|     mVsyncDispatcher->RemoveChildRefreshTimer(this);
 | |
|   }
 | |
|   mVsyncDispatcher = mVsyncSource->GetRefreshTimerVsyncDispatcher();
 | |
|   if (mObservingVsync) {
 | |
|     mVsyncDispatcher->AddChildRefreshTimer(this);
 | |
|   }
 | |
| }
 | |
| 
 | |
| bool VsyncParent::NotifyVsync(const VsyncEvent& aVsync) {
 | |
|   if (IsOnInitialThread()) {
 | |
|     DispatchVsyncEvent(aVsync);
 | |
|     return true;
 | |
|   }
 | |
| 
 | |
|   // Called on hardware vsync thread. We should post to current ipc thread.
 | |
|   nsCOMPtr<nsIRunnable> vsyncEvent = NewRunnableMethod<VsyncEvent>(
 | |
|       "dom::VsyncParent::DispatchVsyncEvent", this,
 | |
|       &VsyncParent::DispatchVsyncEvent, aVsync);
 | |
|   MOZ_ALWAYS_SUCCEEDS(mInitialThread->Dispatch(vsyncEvent, NS_DISPATCH_NORMAL));
 | |
|   return true;
 | |
| }
 | |
| 
 | |
| void VsyncParent::DispatchVsyncEvent(const VsyncEvent& aVsync) {
 | |
|   AssertIsOnInitialThread();
 | |
| 
 | |
|   // If we call NotifyVsync() when we handle ActorDestroy() message, we might
 | |
|   // still call DispatchVsyncEvent().
 | |
|   // Similarly, we might also receive RecvUnobserveVsync() when call
 | |
|   // NotifyVsync(). We use mObservingVsync and mDestroyed flags to skip this
 | |
|   // notification.
 | |
|   if (mObservingVsync && !mDestroyed) {
 | |
|     TimeDuration vsyncRate = mVsyncSource->GetGlobalDisplay().GetVsyncRate();
 | |
|     Unused << SendNotify(aVsync, vsyncRate.ToMilliseconds());
 | |
|   }
 | |
| }
 | |
| 
 | |
| mozilla::ipc::IPCResult VsyncParent::RecvObserve() {
 | |
|   AssertIsOnInitialThread();
 | |
|   if (!mObservingVsync) {
 | |
|     if (mVsyncDispatcher) {
 | |
|       mVsyncDispatcher->AddChildRefreshTimer(this);
 | |
|     }
 | |
|     mObservingVsync = true;
 | |
|     return IPC_OK();
 | |
|   }
 | |
|   return IPC_FAIL_NO_REASON(this);
 | |
| }
 | |
| 
 | |
| mozilla::ipc::IPCResult VsyncParent::RecvUnobserve() {
 | |
|   AssertIsOnInitialThread();
 | |
|   if (mObservingVsync) {
 | |
|     if (mVsyncDispatcher) {
 | |
|       mVsyncDispatcher->RemoveChildRefreshTimer(this);
 | |
|     }
 | |
|     mObservingVsync = false;
 | |
|     return IPC_OK();
 | |
|   }
 | |
|   return IPC_FAIL_NO_REASON(this);
 | |
| }
 | |
| 
 | |
| void VsyncParent::ActorDestroy(ActorDestroyReason aActorDestroyReason) {
 | |
|   MOZ_ASSERT(!mDestroyed);
 | |
|   AssertIsOnInitialThread();
 | |
|   if (mObservingVsync && mVsyncDispatcher) {
 | |
|     mVsyncDispatcher->RemoveChildRefreshTimer(this);
 | |
|   }
 | |
|   mVsyncDispatcher = nullptr;
 | |
|   mDestroyed = true;
 | |
| }
 | |
| 
 | |
| bool VsyncParent::IsOnInitialThread() {
 | |
|   return NS_GetCurrentThread() == mInitialThread;
 | |
| }
 | |
| 
 | |
| void VsyncParent::AssertIsOnInitialThread() { MOZ_ASSERT(IsOnInitialThread()); }
 | |
| 
 | |
| }  // namespace mozilla::dom
 |