gecko-dev/widget/android/nsAppShell.h

251 строка
6.7 KiB
C++

/* -*- Mode: c++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 2; -*- */
/* 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 nsAppShell_h__
#define nsAppShell_h__
#include <time.h>
#include <type_traits>
#include <utility>
#include "mozilla/BackgroundHangMonitor.h"
#include "mozilla/LinkedList.h"
#include "mozilla/Monitor.h"
#include "mozilla/StaticPtr.h"
#include "mozilla/TimeStamp.h" // for mozilla::TimeDuration
#include "mozilla/UniquePtr.h"
#include "mozilla/Unused.h"
#include "mozilla/jni/Natives.h"
#include "nsBaseAppShell.h"
#include "nsCOMPtr.h"
#include "nsIAndroidBridge.h"
#include "nsInterfaceHashtable.h"
#include "nsTArray.h"
namespace mozilla {
bool ProcessNextEvent();
void NotifyEvent();
} // namespace mozilla
class nsWindow;
class nsAppShell : public nsBaseAppShell {
public:
struct Event : mozilla::LinkedListElement<Event> {
static uint64_t GetTime() {
timespec time;
if (clock_gettime(CLOCK_MONOTONIC, &time)) {
return 0ull;
}
return uint64_t(time.tv_sec) * 1000000000ull + time.tv_nsec;
}
uint64_t mPostTime{0};
bool HasSameTypeAs(const Event* other) const {
// Compare vtable addresses to determine same type.
return *reinterpret_cast<const uintptr_t*>(this) ==
*reinterpret_cast<const uintptr_t*>(other);
}
virtual ~Event() {}
virtual void Run() = 0;
virtual void PostTo(mozilla::LinkedList<Event>& queue) {
queue.insertBack(this);
}
virtual bool IsUIEvent() const { return false; }
};
template <typename T>
class LambdaEvent : public Event {
protected:
T lambda;
public:
explicit LambdaEvent(T&& l) : lambda(std::move(l)) {}
void Run() override { lambda(); }
};
class ProxyEvent : public Event {
protected:
mozilla::UniquePtr<Event> baseEvent;
public:
explicit ProxyEvent(mozilla::UniquePtr<Event>&& event)
: baseEvent(std::move(event)) {}
void PostTo(mozilla::LinkedList<Event>& queue) override {
baseEvent->PostTo(queue);
}
void Run() override { baseEvent->Run(); }
};
static nsAppShell* Get() {
MOZ_ASSERT(NS_IsMainThread());
return sAppShell;
}
nsAppShell();
NS_DECL_ISUPPORTS_INHERITED
NS_DECL_NSIOBSERVER
nsresult Init();
void NotifyNativeEvent();
bool ProcessNextNativeEvent(bool mayWait) override;
// Post a subclass of Event.
// e.g. PostEvent(mozilla::MakeUnique<MyEvent>());
template <typename T, typename D>
static void PostEvent(mozilla::UniquePtr<T, D>&& event) {
mozilla::MutexAutoLock lock(*sAppShellLock);
if (!sAppShell) {
return;
}
sAppShell->mEventQueue.Post(std::move(event));
}
// Post a event that will call a lambda
// e.g. PostEvent([=] { /* do something */ });
template <typename T>
static void PostEvent(T&& lambda) {
mozilla::MutexAutoLock lock(*sAppShellLock);
if (!sAppShell) {
return;
}
sAppShell->mEventQueue.Post(
mozilla::MakeUnique<LambdaEvent<T>>(std::move(lambda)));
}
// Post a event and wait for it to finish running on the Gecko thread.
static bool SyncRunEvent(
Event&& event,
mozilla::UniquePtr<Event> (*eventFactory)(mozilla::UniquePtr<Event>&&) =
nullptr,
const mozilla::TimeDuration timeout = mozilla::TimeDuration::Forever());
template <typename T>
static std::enable_if_t<!std::is_base_of<Event, T>::value, void> SyncRunEvent(
T&& lambda) {
SyncRunEvent(LambdaEvent<T>(std::forward<T>(lambda)));
}
static already_AddRefed<nsIURI> ResolveURI(const nsCString& aUriStr);
void SetBrowserApp(nsIAndroidBrowserApp* aBrowserApp) {
mBrowserApp = aBrowserApp;
}
nsIAndroidBrowserApp* GetBrowserApp() { return mBrowserApp; }
protected:
static nsAppShell* sAppShell;
static mozilla::StaticAutoPtr<mozilla::Mutex> sAppShellLock;
static void RecordLatencies();
virtual ~nsAppShell();
nsresult AddObserver(const nsAString& aObserverKey, nsIObserver* aObserver);
class NativeCallbackEvent : public Event {
// Capturing the nsAppShell instance is safe because if the app
// shell is detroyed, this lambda will not be called either.
nsAppShell* const appShell;
public:
explicit NativeCallbackEvent(nsAppShell* as) : appShell(as) {}
void Run() override { appShell->NativeEventCallback(); }
};
void ScheduleNativeEventCallback() override {
mEventQueue.Post(mozilla::MakeUnique<NativeCallbackEvent>(this));
}
class Queue {
private:
mozilla::Monitor mMonitor;
mozilla::LinkedList<Event> mQueue;
public:
enum { LATENCY_UI, LATENCY_OTHER, LATENCY_COUNT };
static uint32_t sLatencyCount[LATENCY_COUNT];
static uint64_t sLatencyTime[LATENCY_COUNT];
Queue() : mMonitor("nsAppShell.Queue") {}
void Signal() {
mozilla::MonitorAutoLock lock(mMonitor);
lock.NotifyAll();
}
void Post(mozilla::UniquePtr<Event>&& event) {
MOZ_ASSERT(event && !event->isInList());
mozilla::MonitorAutoLock lock(mMonitor);
event->PostTo(mQueue);
if (event->isInList()) {
event->mPostTime = Event::GetTime();
// Ownership of event object transfers to the queue.
mozilla::Unused << event.release();
}
lock.NotifyAll();
}
mozilla::UniquePtr<Event> Pop(bool mayWait) {
#ifdef EARLY_BETA_OR_EARLIER
bool isQueueEmpty = false;
if (mayWait) {
mozilla::MonitorAutoLock lock(mMonitor);
isQueueEmpty = mQueue.isEmpty();
}
if (isQueueEmpty) {
// Record latencies when we're about to be idle.
// Note: We can't call this while holding the lock because
// nsAppShell::RecordLatencies may try to dispatch an event to the main
// thread which tries to acquire the lock again.
nsAppShell::RecordLatencies();
}
#endif
mozilla::MonitorAutoLock lock(mMonitor);
if (mayWait && mQueue.isEmpty()) {
lock.Wait();
}
// Ownership of event object transfers to the return value.
mozilla::UniquePtr<Event> event(mQueue.popFirst());
if (!event || !event->mPostTime) {
return event;
}
#ifdef EARLY_BETA_OR_EARLIER
const size_t latencyType =
event->IsUIEvent() ? LATENCY_UI : LATENCY_OTHER;
const uint64_t latency = Event::GetTime() - event->mPostTime;
sLatencyCount[latencyType]++;
sLatencyTime[latencyType] += latency;
#endif
return event;
}
} mEventQueue;
private:
mozilla::CondVar mSyncRunFinished;
bool mSyncRunQuit;
nsCOMPtr<nsIAndroidBrowserApp> mBrowserApp;
nsInterfaceHashtable<nsStringHashKey, nsIObserver> mObserversHash;
};
#endif // nsAppShell_h__