/* -*- 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 nsThread_h__ #define nsThread_h__ #include "mozilla/Mutex.h" #include "nsIThreadInternal.h" #include "nsISupportsPriority.h" #include "nsEventQueue.h" #include "nsThreadUtils.h" #include "nsString.h" #include "nsTObserverArray.h" #include "mozilla/Attributes.h" #include "nsAutoPtr.h" #include "mozilla/AlreadyAddRefed.h" namespace mozilla { class CycleCollectedJSRuntime; } // A native thread class nsThread : public nsIThreadInternal , public nsISupportsPriority { public: NS_DECL_THREADSAFE_ISUPPORTS NS_DECL_NSIEVENTTARGET NS_DECL_NSITHREAD NS_DECL_NSITHREADINTERNAL NS_DECL_NSISUPPORTSPRIORITY // missing from NS_DECL_NSIEVENTTARGET because MSVC nsresult Dispatch(nsIRunnable* aEvent, uint32_t aFlags) { return Dispatch(nsCOMPtr(aEvent).forget(), aFlags); } enum MainThreadFlag { MAIN_THREAD, NOT_MAIN_THREAD }; nsThread(MainThreadFlag aMainThread, uint32_t aStackSize); // Initialize this as a wrapper for a new PRThread. nsresult Init(); // Initialize this as a wrapper for the current PRThread. nsresult InitCurrentThread(); // The PRThread corresponding to this thread. PRThread* GetPRThread() { return mThread; } // If this flag is true, then the nsThread was created using // nsIThreadManager::NewThread. bool ShutdownRequired() { return mShutdownRequired; } // Clear the observer list. void ClearObservers() { mEventObservers.Clear(); } void SetScriptObserver(mozilla::CycleCollectedJSRuntime* aScriptObserver); uint32_t RecursionDepth() const; void ShutdownComplete(struct nsThreadShutdownContext* aContext); void WaitForAllAsynchronousShutdowns(); protected: class nsChainedEventQueue; class nsNestedEventTarget; friend class nsNestedEventTarget; friend class nsThreadShutdownEvent; virtual ~nsThread(); bool ShuttingDown() { return mShutdownContext != nullptr; } static void ThreadFunc(void* aArg); // Helper already_AddRefed GetObserver() { nsIThreadObserver* obs; nsThread::GetObserver(&obs); return already_AddRefed(obs); } // Wrappers for event queue methods: nsresult PutEvent(nsIRunnable* aEvent, nsNestedEventTarget* aTarget); nsresult PutEvent(already_AddRefed&& aEvent, nsNestedEventTarget* aTarget); nsresult DispatchInternal(already_AddRefed&& aEvent, uint32_t aFlags, nsNestedEventTarget* aTarget); struct nsThreadShutdownContext* ShutdownInternal(bool aSync); // Wrapper for nsEventQueue that supports chaining. class nsChainedEventQueue { public: explicit nsChainedEventQueue(mozilla::Mutex& aLock) : mNext(nullptr) , mQueue(aLock) { } bool GetEvent(bool aMayWait, nsIRunnable** aEvent, mozilla::MutexAutoLock& aProofOfLock) { return mQueue.GetEvent(aMayWait, aEvent, aProofOfLock); } void PutEvent(nsIRunnable* aEvent, mozilla::MutexAutoLock& aProofOfLock) { mQueue.PutEvent(aEvent, aProofOfLock); } void PutEvent(already_AddRefed&& aEvent, mozilla::MutexAutoLock& aProofOfLock) { mQueue.PutEvent(mozilla::Move(aEvent), aProofOfLock); } bool HasPendingEvent(mozilla::MutexAutoLock& aProofOfLock) { return mQueue.HasPendingEvent(aProofOfLock); } nsChainedEventQueue* mNext; RefPtr mEventTarget; private: nsEventQueue mQueue; }; class nsNestedEventTarget final : public nsIEventTarget { public: NS_DECL_THREADSAFE_ISUPPORTS NS_DECL_NSIEVENTTARGET nsNestedEventTarget(nsThread* aThread, nsChainedEventQueue* aQueue) : mThread(aThread) , mQueue(aQueue) { } RefPtr mThread; // This is protected by mThread->mLock. nsChainedEventQueue* mQueue; private: ~nsNestedEventTarget() { } }; // This lock protects access to mObserver, mEvents and mEventsAreDoomed. // All of those fields are only modified on the thread itself (never from // another thread). This means that we can avoid holding the lock while // using mObserver and mEvents on the thread itself. When calling PutEvent // on mEvents, we have to hold the lock to synchronize with PopEventQueue. mozilla::Mutex mLock; nsCOMPtr mObserver; mozilla::CycleCollectedJSRuntime* mScriptObserver; // Only accessed on the target thread. nsAutoTObserverArray, 2> mEventObservers; nsChainedEventQueue* mEvents; // never null nsChainedEventQueue mEventsRoot; int32_t mPriority; PRThread* mThread; uint32_t mNestedEventLoopDepth; uint32_t mStackSize; // The shutdown context for ourselves. struct nsThreadShutdownContext* mShutdownContext; // The shutdown contexts for any other threads we've asked to shut down. nsTArray> mRequestedShutdownContexts; bool mShutdownRequired; // Set to true when events posted to this thread will never run. bool mEventsAreDoomed; MainThreadFlag mIsMainThread; }; #if defined(XP_UNIX) && !defined(ANDROID) && !defined(DEBUG) && HAVE_UALARM \ && defined(_GNU_SOURCE) # define MOZ_CANARY extern int sCanaryOutputFD; #endif #endif // nsThread_h__