forked from mirrors/gecko-dev
		
	 6ebcecc30e
			
		
	
	
		6ebcecc30e
		
	
	
	
	
		
			
			We attach it to WorkerPrivate and DOMNavigationTiming so it will be re-used when it should. WorkerPrivate is used in the Performance APIs, Performance Storage Worker, and Event. DOMNavigationTiming is used only in the Performance APIs, but the crucial part is that when the individual DOMNavigationTiming object is re-used, so will the context seed. This in particular came up with the nav2_test_document_open.html Web Platform Test which illustrated the fact that even if you .open() a new document, the performance navigation data is not supposed to change. MozReview-Commit-ID: GIv6biEo2jY --HG-- extra : rebase_source : da2ad8d9d6e0172679c6af14dba72938e9d2012c
		
			
				
	
	
		
			211 lines
		
	
	
	
		
			5.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			211 lines
		
	
	
	
		
			5.4 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 "PerformanceStorageWorker.h"
 | |
| #include "mozilla/dom/WorkerRef.h"
 | |
| #include "mozilla/dom/WorkerRunnable.h"
 | |
| #include "mozilla/dom/WorkerPrivate.h"
 | |
| 
 | |
| namespace mozilla {
 | |
| namespace dom {
 | |
| 
 | |
| class PerformanceProxyData
 | |
| {
 | |
| public:
 | |
|   PerformanceProxyData(UniquePtr<PerformanceTimingData>&& aData,
 | |
|                        const nsAString& aInitiatorType,
 | |
|                        const nsAString& aEntryName)
 | |
|     : mData(Move(aData))
 | |
|     , mInitiatorType(aInitiatorType)
 | |
|     , mEntryName(aEntryName)
 | |
|   {}
 | |
| 
 | |
|   UniquePtr<PerformanceTimingData> mData;
 | |
|   nsString mInitiatorType;
 | |
|   nsString mEntryName;
 | |
| };
 | |
| 
 | |
| namespace {
 | |
| 
 | |
| // Here we use control runnable because this code must be executed also when in
 | |
| // a sync event loop
 | |
| class PerformanceEntryAdder final : public WorkerControlRunnable
 | |
| {
 | |
| public:
 | |
|   PerformanceEntryAdder(WorkerPrivate* aWorkerPrivate,
 | |
|                         PerformanceStorageWorker* aStorage,
 | |
|                         UniquePtr<PerformanceProxyData>&& aData)
 | |
|     : WorkerControlRunnable(aWorkerPrivate, WorkerThreadUnchangedBusyCount)
 | |
|     , mStorage(aStorage)
 | |
|     , mData(Move(aData))
 | |
|   {}
 | |
| 
 | |
|   bool
 | |
|   WorkerRun(JSContext* aCx, WorkerPrivate* aWorkerPrivate) override
 | |
|   {
 | |
|     mStorage->AddEntryOnWorker(Move(mData));
 | |
|     return true;
 | |
|   }
 | |
| 
 | |
|   nsresult
 | |
|   Cancel() override
 | |
|   {
 | |
|     mStorage->ShutdownOnWorker();
 | |
|     return WorkerRunnable::Cancel();
 | |
|   }
 | |
| 
 | |
|   bool
 | |
|   PreDispatch(WorkerPrivate* aWorkerPrivate) override
 | |
|   {
 | |
|     return true;
 | |
|   }
 | |
| 
 | |
|   void
 | |
|   PostDispatch(WorkerPrivate* aWorkerPrivate, bool aDispatchResult) override
 | |
|   {}
 | |
| 
 | |
| private:
 | |
|   RefPtr<PerformanceStorageWorker> mStorage;
 | |
|   UniquePtr<PerformanceProxyData> mData;
 | |
| };
 | |
| 
 | |
| class PerformanceStorageWorkerHolder final : public WorkerHolder
 | |
| {
 | |
|   RefPtr<PerformanceStorageWorker> mStorage;
 | |
| 
 | |
| public:
 | |
|   explicit PerformanceStorageWorkerHolder(PerformanceStorageWorker* aStorage)
 | |
|     : WorkerHolder("PerformanceStorageWorkerHolder",
 | |
|                    WorkerHolder::AllowIdleShutdownStart)
 | |
|     , mStorage(aStorage)
 | |
|   {}
 | |
| 
 | |
|   bool
 | |
|   Notify(WorkerStatus aStatus) override
 | |
|   {
 | |
|     if (mStorage) {
 | |
|       RefPtr<PerformanceStorageWorker> storage;
 | |
|       storage.swap(mStorage);
 | |
|       storage->ShutdownOnWorker();
 | |
|     }
 | |
| 
 | |
|     return true;
 | |
|   }
 | |
| };
 | |
| 
 | |
| } // anonymous
 | |
| 
 | |
| /* static */ already_AddRefed<PerformanceStorageWorker>
 | |
| PerformanceStorageWorker::Create(WorkerPrivate* aWorkerPrivate)
 | |
| {
 | |
|   MOZ_ASSERT(aWorkerPrivate);
 | |
|   aWorkerPrivate->AssertIsOnWorkerThread();
 | |
| 
 | |
|   RefPtr<PerformanceStorageWorker> storage = new PerformanceStorageWorker();
 | |
| 
 | |
|   storage->mWorkerRef = WeakWorkerRef::Create(aWorkerPrivate, [storage]() {
 | |
|     storage->ShutdownOnWorker();
 | |
|   });
 | |
| 
 | |
|   // PerformanceStorageWorker is created at the creation time of the worker.
 | |
|   MOZ_ASSERT(storage->mWorkerRef);
 | |
| 
 | |
|   return storage.forget();
 | |
| }
 | |
| 
 | |
| PerformanceStorageWorker::PerformanceStorageWorker()
 | |
|   : mMutex("PerformanceStorageWorker::mMutex")
 | |
| {
 | |
| }
 | |
| 
 | |
| PerformanceStorageWorker::~PerformanceStorageWorker() = default;
 | |
| 
 | |
| void
 | |
| PerformanceStorageWorker::AddEntry(nsIHttpChannel* aChannel,
 | |
|                                    nsITimedChannel* aTimedChannel)
 | |
| {
 | |
|   MOZ_ASSERT(NS_IsMainThread());
 | |
| 
 | |
|   MutexAutoLock lock(mMutex);
 | |
| 
 | |
|   if (!mWorkerRef) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   // If we have mWorkerRef, we haven't received the WorkerRef notification and
 | |
|   // we haven't yet call ShutdownOnWorker, which uses the mutex.
 | |
|   WorkerPrivate* workerPrivate = mWorkerRef->GetUnsafePrivate();
 | |
|   MOZ_ASSERT(workerPrivate);
 | |
| 
 | |
|   nsAutoString initiatorType;
 | |
|   nsAutoString entryName;
 | |
| 
 | |
|   UniquePtr<PerformanceTimingData> performanceTimingData(
 | |
|     PerformanceTimingData::Create(aTimedChannel, aChannel, 0, initiatorType,
 | |
|                                   entryName));
 | |
|   if (!performanceTimingData) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   UniquePtr<PerformanceProxyData> data(
 | |
|     new PerformanceProxyData(Move(performanceTimingData), initiatorType,
 | |
|                              entryName));
 | |
| 
 | |
|   RefPtr<PerformanceEntryAdder> r =
 | |
|     new PerformanceEntryAdder(workerPrivate, this, Move(data));
 | |
|   Unused << NS_WARN_IF(!r->Dispatch());
 | |
| }
 | |
| 
 | |
| void
 | |
| PerformanceStorageWorker::ShutdownOnWorker()
 | |
| {
 | |
|   MutexAutoLock lock(mMutex);
 | |
| 
 | |
|   if (!mWorkerRef) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   MOZ_ASSERT(IsCurrentThreadRunningWorker());
 | |
| 
 | |
|   mWorkerRef = nullptr;
 | |
| }
 | |
| 
 | |
| void
 | |
| PerformanceStorageWorker::AddEntryOnWorker(UniquePtr<PerformanceProxyData>&& aData)
 | |
| {
 | |
|   RefPtr<Performance> performance;
 | |
|   UniquePtr<PerformanceProxyData> data = Move(aData);
 | |
| 
 | |
|   {
 | |
|     MutexAutoLock lock(mMutex);
 | |
| 
 | |
|     if (!mWorkerRef) {
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     // We must have the workerPrivate because it is available until a notification
 | |
|     // is received by WorkerRef and we use mutex to make the code protected.
 | |
|     WorkerPrivate* workerPrivate = mWorkerRef->GetPrivate();
 | |
|     MOZ_ASSERT(workerPrivate);
 | |
| 
 | |
|     WorkerGlobalScope* scope = workerPrivate->GlobalScope();
 | |
|     performance = scope->GetPerformance();
 | |
|   }
 | |
| 
 | |
|   if (NS_WARN_IF(!performance)) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   RefPtr<PerformanceResourceTiming> performanceEntry =
 | |
|     new PerformanceResourceTiming(Move(data->mData), performance,
 | |
|                                  data->mEntryName);
 | |
|   performanceEntry->SetInitiatorType(data->mInitiatorType);
 | |
| 
 | |
|   performance->InsertResourceEntry(performanceEntry);
 | |
| }
 | |
| 
 | |
| } // namespace dom
 | |
| } // namespace mozilla
 |