/* -*- 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" // 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(); } static nsresult SetMainThreadObserver(nsIThreadObserver* aObserver); protected: static nsIThreadObserver* sMainThreadObserver; 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: bool GetEvent(bool aMayWait, nsIRunnable** aEvent) { return mEvents->GetEvent(aMayWait, aEvent); } nsresult PutEvent(nsIRunnable* aEvent, nsNestedEventTarget* aTarget); nsresult PutEvent(already_AddRefed&& aEvent, nsNestedEventTarget* aTarget); nsresult DispatchInternal(already_AddRefed&& aEvent, uint32_t aFlags, nsNestedEventTarget* aTarget); // Wrapper for nsEventQueue that supports chaining. class nsChainedEventQueue { public: nsChainedEventQueue() : mNext(nullptr) { } bool GetEvent(bool aMayWait, nsIRunnable** aEvent) { return mQueue.GetEvent(aMayWait, aEvent); } void PutEvent(nsIRunnable* aEvent) { mQueue.PutEvent(aEvent); } void PutEvent(already_AddRefed&& aEvent) { mQueue.PutEvent(mozilla::Move(aEvent)); } bool HasPendingEvent() { return mQueue.HasPendingEvent(); } nsChainedEventQueue* mNext; nsRefPtr 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) { } nsRefPtr 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; // 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; struct nsThreadShutdownContext* mShutdownContext; bool mShutdownRequired; // Set to true when events posted to this thread will never run. bool mEventsAreDoomed; MainThreadFlag mIsMainThread; }; //----------------------------------------------------------------------------- class nsThreadSyncDispatch : public nsRunnable { public: nsThreadSyncDispatch(nsIThread* aOrigin, already_AddRefed&& aTask) : mOrigin(aOrigin) , mSyncTask(aTask) , mResult(NS_ERROR_NOT_INITIALIZED) { } bool IsPending() { return mSyncTask != nullptr; } nsresult Result() { return mResult; } private: NS_DECL_NSIRUNNABLE nsCOMPtr mOrigin; nsCOMPtr mSyncTask; nsresult mResult; }; #if defined(XP_UNIX) && !defined(ANDROID) && !defined(DEBUG) && HAVE_UALARM \ && defined(_GNU_SOURCE) # define MOZ_CANARY extern int sCanaryOutputFD; #endif #endif // nsThread_h__