/* -*- 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/. */ #ifndef mozilla_dom_workers_runtimeservice_h__ #define mozilla_dom_workers_runtimeservice_h__ #include "Workers.h" #include "nsIObserver.h" #include "mozilla/dom/BindingDeclarations.h" #include "nsClassHashtable.h" #include "nsHashKeys.h" #include "nsTArray.h" class nsITimer; class nsPIDOMWindow; BEGIN_WORKERS_NAMESPACE class SharedWorker; class WorkerThread; class RuntimeService final : public nsIObserver { struct SharedWorkerInfo { WorkerPrivate* mWorkerPrivate; nsCString mScriptSpec; nsCString mName; SharedWorkerInfo(WorkerPrivate* aWorkerPrivate, const nsACString& aScriptSpec, const nsACString& aName) : mWorkerPrivate(aWorkerPrivate), mScriptSpec(aScriptSpec), mName(aName) { } }; struct WorkerDomainInfo { nsCString mDomain; nsTArray mActiveWorkers; nsTArray mActiveServiceWorkers; nsTArray mQueuedWorkers; nsClassHashtable mSharedWorkerInfos; uint32_t mChildWorkerCount; WorkerDomainInfo() : mActiveWorkers(1), mChildWorkerCount(0) { } uint32_t ActiveWorkerCount() const { return mActiveWorkers.Length() + mChildWorkerCount; } uint32_t ActiveServiceWorkerCount() const { return mActiveServiceWorkers.Length(); } bool HasNoWorkers() const { return ActiveWorkerCount() == 0 && ActiveServiceWorkerCount() == 0; } }; struct IdleThreadInfo; mozilla::Mutex mMutex; // Protected by mMutex. nsClassHashtable mDomainMap; // Protected by mMutex. nsTArray mIdleThreadArray; // *Not* protected by mMutex. nsClassHashtable, nsTArray > mWindowMap; // Only used on the main thread. nsCOMPtr mIdleThreadTimer; static JSSettings sDefaultJSSettings; static bool sDefaultPreferences[WORKERPREF_COUNT]; public: struct NavigatorProperties { nsString mAppName; nsString mAppNameOverridden; nsString mAppVersion; nsString mAppVersionOverridden; nsString mPlatform; nsString mPlatformOverridden; nsTArray mLanguages; }; private: NavigatorProperties mNavigatorProperties; // True when the observer service holds a reference to this object. bool mObserved; bool mShuttingDown; bool mNavigatorPropertiesLoaded; public: NS_DECL_ISUPPORTS NS_DECL_NSIOBSERVER static RuntimeService* GetOrCreateService(); static RuntimeService* GetService(); bool RegisterWorker(JSContext* aCx, WorkerPrivate* aWorkerPrivate); void UnregisterWorker(JSContext* aCx, WorkerPrivate* aWorkerPrivate); void RemoveSharedWorker(WorkerDomainInfo* aDomainInfo, WorkerPrivate* aWorkerPrivate); void CancelWorkersForWindow(nsPIDOMWindow* aWindow); void FreezeWorkersForWindow(nsPIDOMWindow* aWindow); void ThawWorkersForWindow(nsPIDOMWindow* aWindow); void SuspendWorkersForWindow(nsPIDOMWindow* aWindow); void ResumeWorkersForWindow(nsPIDOMWindow* aWindow); nsresult CreateSharedWorker(const GlobalObject& aGlobal, const nsAString& aScriptURL, const nsACString& aName, SharedWorker** aSharedWorker); void ForgetSharedWorker(WorkerPrivate* aWorkerPrivate); const NavigatorProperties& GetNavigatorProperties() const { return mNavigatorProperties; } void NoteIdleThread(WorkerThread* aThread); static void GetDefaultJSSettings(JSSettings& aSettings) { AssertIsOnMainThread(); aSettings = sDefaultJSSettings; } static void GetDefaultPreferences(bool aPreferences[WORKERPREF_COUNT]) { AssertIsOnMainThread(); memcpy(aPreferences, sDefaultPreferences, WORKERPREF_COUNT * sizeof(bool)); } static void SetDefaultRuntimeOptions(const JS::RuntimeOptions& aRuntimeOptions) { AssertIsOnMainThread(); sDefaultJSSettings.runtimeOptions = aRuntimeOptions; } void UpdateAppNameOverridePreference(const nsAString& aValue); void UpdateAppVersionOverridePreference(const nsAString& aValue); void UpdatePlatformOverridePreference(const nsAString& aValue); void UpdateAllWorkerRuntimeOptions(); void UpdateAllWorkerLanguages(const nsTArray& aLanguages); void UpdateAllWorkerPreference(WorkerPreference aPref, bool aValue); static void SetDefaultJSGCSettings(JSGCParamKey aKey, uint32_t aValue) { AssertIsOnMainThread(); sDefaultJSSettings.ApplyGCSetting(aKey, aValue); } void UpdateAllWorkerMemoryParameter(JSGCParamKey aKey, uint32_t aValue); static uint32_t GetContentCloseHandlerTimeoutSeconds() { return sDefaultJSSettings.content.maxScriptRuntime; } static uint32_t GetChromeCloseHandlerTimeoutSeconds() { return sDefaultJSSettings.chrome.maxScriptRuntime; } #ifdef JS_GC_ZEAL static void SetDefaultGCZeal(uint8_t aGCZeal, uint32_t aFrequency) { AssertIsOnMainThread(); sDefaultJSSettings.gcZeal = aGCZeal; sDefaultJSSettings.gcZealFrequency = aFrequency; } void UpdateAllWorkerGCZeal(); #endif void GarbageCollectAllWorkers(bool aShrinking); void CycleCollectAllWorkers(); void SendOfflineStatusChangeEventToAllWorkers(bool aIsOffline); private: RuntimeService(); ~RuntimeService(); nsresult Init(); void Shutdown(); void Cleanup(); void AddAllTopLevelWorkersToArray(nsTArray& aWorkers); void GetWorkersForWindow(nsPIDOMWindow* aWindow, nsTArray& aWorkers); bool ScheduleWorker(JSContext* aCx, WorkerPrivate* aWorkerPrivate); static void ShutdownIdleThreads(nsITimer* aTimer, void* aClosure); static void WorkerPrefChanged(const char* aPrefName, void* aClosure); static void JSVersionChanged(const char* aPrefName, void* aClosure); nsresult CreateSharedWorkerFromLoadInfo(JSContext* aCx, WorkerLoadInfo* aLoadInfo, const nsAString& aScriptURL, const nsACString& aName, SharedWorker** aSharedWorker); }; END_WORKERS_NAMESPACE #endif /* mozilla_dom_workers_runtimeservice_h__ */