/* -*- 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 "WakeLockJS.h" #include "ErrorList.h" #include "WakeLock.h" #include "WakeLockSentinel.h" #include "mozilla/AlreadyAddRefed.h" #include "mozilla/Assertions.h" #include "mozilla/Hal.h" #include "mozilla/Logging.h" #include "mozilla/StaticPrefs_dom.h" #include "mozilla/dom/Document.h" #include "mozilla/dom/Event.h" #include "mozilla/dom/EventTarget.h" #include "mozilla/dom/FeaturePolicyUtils.h" #include "mozilla/dom/Navigator.h" #include "mozilla/dom/Promise.h" #include "mozilla/dom/WakeLockBinding.h" #include "nsCOMPtr.h" #include "nsCRT.h" #include "nsContentPermissionHelper.h" #include "nsError.h" #include "nsIGlobalObject.h" #include "nsISupports.h" #include "nsPIDOMWindow.h" #include "nsServiceManagerUtils.h" #include "nscore.h" namespace mozilla::dom { static mozilla::LazyLogModule sLogger("ScreenWakeLock"); #define MIN_BATTERY_LEVEL 0.05 nsLiteralCString WakeLockJS::GetRequestErrorMessage(RequestError aRv) { switch (aRv) { case RequestError::DocInactive: return "The requesting document is inactive."_ns; case RequestError::DocHidden: return "The requesting document is hidden."_ns; case RequestError::PolicyDisallowed: return "A permissions policy does not allow screen-wake-lock for the requesting document."_ns; case RequestError::PrefDisabled: return "The pref dom.screenwakelock.enabled is disabled."_ns; case RequestError::InternalFailure: return "A browser-internal error occured."_ns; case RequestError::PermissionDenied: return "Permission to request screen-wake-lock was denied."_ns; default: MOZ_ASSERT_UNREACHABLE("Unknown error reason"); return "Unknown error"_ns; } } // https://w3c.github.io/screen-wake-lock/#the-request-method steps 2-5 WakeLockJS::RequestError WakeLockJS::WakeLockAllowedForDocument( Document* aDoc) { if (!aDoc) { return RequestError::InternalFailure; } // Step 2. check policy-controlled feature screen-wake-lock if (!FeaturePolicyUtils::IsFeatureAllowed(aDoc, u"screen-wake-lock"_ns)) { return RequestError::PolicyDisallowed; } // Step 3. Deny wake lock for user agent specific reasons if (!StaticPrefs::dom_screenwakelock_enabled()) { return RequestError::PrefDisabled; } // Step 4 check doc active if (!aDoc->IsActive()) { return RequestError::DocInactive; } // Step 5. check doc visible if (aDoc->Hidden()) { return RequestError::DocHidden; } return RequestError::Success; } // https://w3c.github.io/screen-wake-lock/#dfn-applicable-wake-lock static bool IsWakeLockApplicable(WakeLockType aType) { hal::BatteryInformation batteryInfo; hal::GetCurrentBatteryInformation(&batteryInfo); if (batteryInfo.level() <= MIN_BATTERY_LEVEL && !batteryInfo.charging()) { return false; } // only currently supported wake lock type return aType == WakeLockType::Screen; } // https://w3c.github.io/screen-wake-lock/#dfn-release-a-wake-lock void ReleaseWakeLock(Document* aDoc, WakeLockSentinel* aLock, WakeLockType aType) { MOZ_ASSERT(aLock); MOZ_ASSERT(aDoc); RefPtr kungFuDeathGrip = aLock; aDoc->ActiveWakeLocks(aType).Remove(aLock); aLock->NotifyLockReleased(); MOZ_LOG(sLogger, LogLevel::Debug, ("Released wake lock sentinel")); } NS_IMPL_CYCLE_COLLECTION_WRAPPERCACHE_CLASS(WakeLockJS) NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN(WakeLockJS) NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mWindow) NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END NS_IMPL_CYCLE_COLLECTION_UNLINK_BEGIN(WakeLockJS) tmp->DetachListeners(); NS_IMPL_CYCLE_COLLECTION_UNLINK(mWindow) NS_IMPL_CYCLE_COLLECTION_UNLINK_PRESERVED_WRAPPER NS_IMPL_CYCLE_COLLECTION_UNLINK_END NS_IMPL_CYCLE_COLLECTING_ADDREF(WakeLockJS) NS_IMPL_CYCLE_COLLECTING_RELEASE(WakeLockJS) NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION(WakeLockJS) NS_WRAPPERCACHE_INTERFACE_MAP_ENTRY NS_INTERFACE_MAP_ENTRY_AMBIGUOUS(nsISupports, nsIObserver) NS_INTERFACE_MAP_ENTRY(nsIObserver) NS_INTERFACE_MAP_ENTRY(nsISupportsWeakReference) NS_INTERFACE_MAP_END WakeLockJS::WakeLockJS(nsPIDOMWindowInner* aWindow) : mWindow(aWindow) { AttachListeners(); } WakeLockJS::~WakeLockJS() { DetachListeners(); } nsISupports* WakeLockJS::GetParentObject() const { return mWindow; } JSObject* WakeLockJS::WrapObject(JSContext* aCx, JS::Handle aGivenProto) { return WakeLock_Binding::Wrap(aCx, this, aGivenProto); } // https://w3c.github.io/screen-wake-lock/#the-request-method Step 7.3 Result, WakeLockJS::RequestError> WakeLockJS::Obtain(WakeLockType aType, Document* aDoc) { // Step 7.3.1. check visibility again // Out of spec, but also check everything else RequestError rv = WakeLockAllowedForDocument(aDoc); if (rv != RequestError::Success) { return Err(rv); } // Step 7.3.3. let lock be a new WakeLockSentinel RefPtr lock = MakeRefPtr(mWindow->AsGlobal(), aType); // Step 7.3.2. acquire a wake lock if (IsWakeLockApplicable(aType)) { lock->AcquireActualLock(); } // Steps 7.3.4. append lock to locks aDoc->ActiveWakeLocks(aType).Insert(lock); return lock.forget(); } // https://w3c.github.io/screen-wake-lock/#the-request-method already_AddRefed WakeLockJS::Request(WakeLockType aType, ErrorResult& aRv) { MOZ_LOG(sLogger, LogLevel::Debug, ("Received request for wake lock")); nsCOMPtr global = mWindow->AsGlobal(); RefPtr promise = Promise::Create(global, aRv); NS_ENSURE_FALSE(aRv.Failed(), nullptr); // Steps 1-5 nsCOMPtr doc = mWindow->GetExtantDoc(); RequestError rv = WakeLockAllowedForDocument(doc); if (rv != RequestError::Success) { promise->MaybeRejectWithNotAllowedError(GetRequestErrorMessage(rv)); return promise.forget(); } // For now, we don't check the permission as we always grant the lock // Step 7.3. Queue a task NS_DispatchToMainThread(NS_NewRunnableFunction( "ObtainWakeLock", [aType, promise, doc, self = RefPtr(this)]() { auto lockOrErr = self->Obtain(aType, doc); if (lockOrErr.isOk()) { RefPtr lock = lockOrErr.unwrap(); promise->MaybeResolve(lock); MOZ_LOG(sLogger, LogLevel::Debug, ("Resolved promise with wake lock sentinel")); } else { promise->MaybeRejectWithNotAllowedError( GetRequestErrorMessage(lockOrErr.unwrapErr())); } })); return promise.forget(); } void WakeLockJS::AttachListeners() { hal::RegisterBatteryObserver(this); nsCOMPtr prefBranch = do_GetService(NS_PREFSERVICE_CONTRACTID); MOZ_ASSERT(prefBranch); DebugOnly rv = prefBranch->AddObserver("dom.screenwakelock.enabled", this, true); MOZ_ASSERT(NS_SUCCEEDED(rv)); } void WakeLockJS::DetachListeners() { hal::UnregisterBatteryObserver(this); if (nsCOMPtr prefBranch = do_GetService(NS_PREFSERVICE_CONTRACTID)) { prefBranch->RemoveObserver("dom.screenwakelock.enabled", this); } } NS_IMETHODIMP WakeLockJS::Observe(nsISupports* aSubject, const char* aTopic, const char16_t* aData) { if (nsCRT::strcmp(aTopic, NS_PREFBRANCH_PREFCHANGE_TOPIC_ID) == 0) { if (!StaticPrefs::dom_screenwakelock_enabled()) { nsCOMPtr doc = mWindow->GetExtantDoc(); MOZ_ASSERT(doc); doc->UnlockAllWakeLocks(WakeLockType::Screen); } } return NS_OK; } void WakeLockJS::Notify(const hal::BatteryInformation& aBatteryInfo) { if (aBatteryInfo.level() > MIN_BATTERY_LEVEL || aBatteryInfo.charging()) { return; } nsCOMPtr doc = mWindow->GetExtantDoc(); MOZ_ASSERT(doc); doc->UnlockAllWakeLocks(WakeLockType::Screen); } } // namespace mozilla::dom