mirror of
https://github.com/roytam1/palemoon27.git
synced 2026-05-26 14:18:48 +00:00
15db8f16bf
- Bug 1132854 - Remove the gfx::ToIntSize conversion helper. r=Bas (0ee451e53) - Bug 1020179 - Let PContent manage PContentPermissionRequest. r=fabrice, r=khuey (caf96b54e) - Bug 1153023 - Convert TabParent::mChromeOffset to a LayoutDeviceIntPoint. r=billm (ba338584a) - Bug 1139033 - Don't schedule an unnecessary repeat transaction when doing a non-progressive paint. r=nical (9c77d9318) - Bug 1137203 - Ignore the critical displayport when a layer is subject to OMTA relative to the scrolling ancestor. r=BenWa (c9bedfb1e) - Bug 1137203 - Cleanup to ditch the fast-path code entirely and just prevent progressive drawing in the equivalent scenarios. r=BenWa (6bc5b8813) - Bug 1128042 - Don't round critical displayport out as it should already be tile aligned and rounding error can increase tile usage. r=botond (03e34e2c5) - Bug 1149461 - Disable progressive drawing unless the compositor is actively scrolling a tiled layer. r=nical (7a3de28cb) - Bug 1152838 - Correctly inflate valid regions to tile boundaries. r=mattwoodrow (7a496f645) - Bug 1148971 - Make nsITheme::GetMinimumWidgetSize return a LayoutDeviceIntSize result instead of the unit-less nsIntSize type. r=roc (c34ddc478) - Bug 1155621 - Make nsIntRect and nsIntPoint typedefs of mozilla::gfx::IntRect and mozilla::gfx::IntPoint. r=Bas (9901a37f6) - Bug 1156632 - Remove unused forward class declarations - patch 5 - rdf, parser, layout and something else, r=ehsan (794739bd3) - Bug 1156632 - Remove unused forward class declarations - patch 6 - the rest of the tree, r=ehsan (63a2c4cb4) - Bug 1161634 - Allow synthesizing native mouse-scroll events on Linux. r=karlt (b88c1f367) - Bug 1161634 - Enable the test_wheel_scroll on Linux as well. r=mstange (7dab62ad4) - Bug 1144643 - Render tooltips as transparent on Gtk3. r=karlt (80f7fa312) - Bug 1174966 part 1 - Change type of mCancelledPointerLockRequests field from uint32_t to bit field. r=smaug (ef235c33e) - Bug 1155030 - Fix asterix/asterisk misspelling. r=ehsan (edb769304) - Bug 1149194 - Don't use uninitialized value in ComputedTimingFunction::operator==. r=bbirtles (6bc804d45) - Bug 1151346 - Make ActiveLayerTracker::IsOffsetOrMarginStyleAnimated respect CSS animations. r=roc (2c3f24ba0) - Bug 1151346 - Back out the important part again because of bug 1151889. (002b0e67b) - Bug 1122414 part 1 - Factor out a TransitionProperty method in ElementPropertyTransition; r=jwatt (ddbbdb04c) - Bug 1122414 part 2 - Return the transitionProperty from Animation.name for CSS transitions; r=jwatt (689251b93) - Bug 1122414 part 3 - Update DevTools tests to expect a name for transitions; r=pbrosset (f0d7e57e9) - Bug 1149999 - 1 - Display transition names in animation-panel now that they have names; r=past (ea3d8d552) - Bug 1120343 - 1 - Allow setting animations' currentTime by clicking/dragging the timeline; r=miker (936996d21) - Bug 1120343 - 2 - Add rewind and fast-forward buttons to animation player widgets; r=miker (95eddc465) - Bug 1120343 - 3 - Tests for the current time control in the animation panel; r=miker (b8a93f858) - Bug 1110762 - Add a setCurrentTime method to the animation actor; r=past (d0dae8967) - Bug 1123851 - 1 - Element geometry highlighter; r=bgrins (89b1a83bf) - Bug 1123851 - 2 - Tests for the element geometry highlighter; r=bgrins (7542bcab0) - add missing part of accessing first element from Bug 1139925 - Make the BoxModelHighlighter highlight all quads (b5c6076c1) - Bug 1139186 - 1 - Refactor to the native anon nodes manipulation in highlighters; r=bgrins (a454aefbf) - Bug 1139186 - 2 - Add event handling support to CanvasFrameAnonymousContentHelper; r=bgrins (a705c2716) - Bug 1120339 - Add setPlaybackRate method to AnimationPlayerActor; r=past (efa167a19) - Bug 1120833 - 2 - Fire events about changed animations in the AnimationsActor; r=past (4b5fddd23) - Bug 1120833 - 1 - Remove EventEmitter usage in AnimationPlayerFront (21186e616)
456 lines
13 KiB
C++
456 lines
13 KiB
C++
/* 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 "DOMCameraManager.h"
|
|
#include "nsDebug.h"
|
|
#include "jsapi.h"
|
|
#include "Navigator.h"
|
|
#include "nsPIDOMWindow.h"
|
|
#include "mozilla/Services.h"
|
|
#include "nsContentPermissionHelper.h"
|
|
#include "nsIContentPermissionPrompt.h"
|
|
#include "nsIObserverService.h"
|
|
#include "nsIPermissionManager.h"
|
|
#include "nsIScriptObjectPrincipal.h"
|
|
#include "DOMCameraControl.h"
|
|
#include "nsDOMClassInfo.h"
|
|
#include "CameraCommon.h"
|
|
#include "CameraPreferences.h"
|
|
#include "mozilla/dom/BindingUtils.h"
|
|
#include "mozilla/dom/PermissionMessageUtils.h"
|
|
#include "nsQueryObject.h"
|
|
|
|
using namespace mozilla;
|
|
using namespace mozilla::dom;
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_WRAPPERCACHE(nsDOMCameraManager, mWindow)
|
|
|
|
NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION(nsDOMCameraManager)
|
|
NS_WRAPPERCACHE_INTERFACE_MAP_ENTRY
|
|
NS_INTERFACE_MAP_ENTRY_AMBIGUOUS(nsISupports, nsIObserver)
|
|
NS_INTERFACE_MAP_ENTRY(nsISupportsWeakReference)
|
|
NS_INTERFACE_MAP_ENTRY(nsIObserver)
|
|
NS_INTERFACE_MAP_END
|
|
|
|
NS_IMPL_CYCLE_COLLECTING_ADDREF(nsDOMCameraManager)
|
|
NS_IMPL_CYCLE_COLLECTING_RELEASE(nsDOMCameraManager)
|
|
|
|
/**
|
|
* Global camera logging object
|
|
*
|
|
* Set the NSPR_LOG_MODULES environment variable to enable logging
|
|
* in a debug build, e.g. NSPR_LOG_MODULES=Camera:5
|
|
*/
|
|
PRLogModuleInfo*
|
|
GetCameraLog()
|
|
{
|
|
static PRLogModuleInfo *sLog;
|
|
if (!sLog) {
|
|
sLog = PR_NewLogModule("Camera");
|
|
}
|
|
return sLog;
|
|
}
|
|
|
|
::WindowTable* nsDOMCameraManager::sActiveWindows = nullptr;
|
|
|
|
nsDOMCameraManager::nsDOMCameraManager(nsPIDOMWindow* aWindow)
|
|
: mWindowId(aWindow->WindowID())
|
|
, mPermission(nsIPermissionManager::DENY_ACTION)
|
|
, mWindow(aWindow)
|
|
{
|
|
/* member initializers and constructor code */
|
|
DOM_CAMERA_LOGT("%s:%d : this=%p, windowId=%" PRIx64 "\n", __func__, __LINE__, this, mWindowId);
|
|
MOZ_COUNT_CTOR(nsDOMCameraManager);
|
|
}
|
|
|
|
nsDOMCameraManager::~nsDOMCameraManager()
|
|
{
|
|
/* destructor code */
|
|
MOZ_COUNT_DTOR(nsDOMCameraManager);
|
|
DOM_CAMERA_LOGT("%s:%d : this=%p\n", __func__, __LINE__, this);
|
|
}
|
|
|
|
/* static */
|
|
void
|
|
nsDOMCameraManager::GetListOfCameras(nsTArray<nsString>& aList, ErrorResult& aRv)
|
|
{
|
|
aRv = ICameraControl::GetListOfCameras(aList);
|
|
}
|
|
|
|
/* static */
|
|
bool
|
|
nsDOMCameraManager::HasSupport(JSContext* aCx, JSObject* aGlobal)
|
|
{
|
|
return Navigator::HasCameraSupport(aCx, aGlobal);
|
|
}
|
|
|
|
/* static */
|
|
bool
|
|
nsDOMCameraManager::CheckPermission(nsPIDOMWindow* aWindow)
|
|
{
|
|
nsCOMPtr<nsIPermissionManager> permMgr =
|
|
services::GetPermissionManager();
|
|
NS_ENSURE_TRUE(permMgr, false);
|
|
|
|
uint32_t permission = nsIPermissionManager::DENY_ACTION;
|
|
permMgr->TestPermissionFromWindow(aWindow, "camera", &permission);
|
|
if (permission != nsIPermissionManager::ALLOW_ACTION &&
|
|
permission != nsIPermissionManager::PROMPT_ACTION) {
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/* static */
|
|
already_AddRefed<nsDOMCameraManager>
|
|
nsDOMCameraManager::CreateInstance(nsPIDOMWindow* aWindow)
|
|
{
|
|
// Initialize the shared active window tracker
|
|
if (!sActiveWindows) {
|
|
sActiveWindows = new ::WindowTable();
|
|
}
|
|
|
|
nsRefPtr<nsDOMCameraManager> cameraManager =
|
|
new nsDOMCameraManager(aWindow);
|
|
|
|
nsCOMPtr<nsIObserverService> obs = services::GetObserverService();
|
|
if (!obs) {
|
|
DOM_CAMERA_LOGE("Camera manager failed to get observer service\n");
|
|
return nullptr;
|
|
}
|
|
|
|
nsresult rv = obs->AddObserver(cameraManager, "xpcom-shutdown", true);
|
|
if (NS_FAILED(rv)) {
|
|
DOM_CAMERA_LOGE("Camera manager failed to add 'xpcom-shutdown' observer (0x%x)\n", rv);
|
|
return nullptr;
|
|
}
|
|
|
|
return cameraManager.forget();
|
|
}
|
|
|
|
class CameraPermissionRequest : public nsIContentPermissionRequest
|
|
, public nsIRunnable
|
|
{
|
|
public:
|
|
NS_DECL_CYCLE_COLLECTING_ISUPPORTS
|
|
NS_DECL_NSICONTENTPERMISSIONREQUEST
|
|
NS_DECL_NSIRUNNABLE
|
|
NS_DECL_CYCLE_COLLECTION_CLASS_AMBIGUOUS(CameraPermissionRequest,
|
|
nsIContentPermissionRequest)
|
|
|
|
CameraPermissionRequest(nsIPrincipal* aPrincipal,
|
|
nsPIDOMWindow* aWindow,
|
|
nsRefPtr<nsDOMCameraManager> aManager,
|
|
uint32_t aCameraId,
|
|
const CameraConfiguration& aInitialConfig,
|
|
nsRefPtr<Promise> aPromise)
|
|
: mPrincipal(aPrincipal)
|
|
, mWindow(aWindow)
|
|
, mCameraManager(aManager)
|
|
, mCameraId(aCameraId)
|
|
, mInitialConfig(aInitialConfig)
|
|
, mPromise(aPromise)
|
|
, mRequester(new nsContentPermissionRequester(mWindow))
|
|
{ }
|
|
|
|
protected:
|
|
virtual ~CameraPermissionRequest() { }
|
|
|
|
nsresult DispatchCallback(uint32_t aPermission);
|
|
void CallAllow();
|
|
void CallCancel();
|
|
nsCOMPtr<nsIPrincipal> mPrincipal;
|
|
nsCOMPtr<nsPIDOMWindow> mWindow;
|
|
nsRefPtr<nsDOMCameraManager> mCameraManager;
|
|
uint32_t mCameraId;
|
|
CameraConfiguration mInitialConfig;
|
|
nsRefPtr<Promise> mPromise;
|
|
nsCOMPtr<nsIContentPermissionRequester> mRequester;
|
|
};
|
|
|
|
NS_IMPL_CYCLE_COLLECTION(CameraPermissionRequest, mWindow, mPromise)
|
|
|
|
NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION(CameraPermissionRequest)
|
|
NS_INTERFACE_MAP_ENTRY(nsIContentPermissionRequest)
|
|
NS_INTERFACE_MAP_ENTRY(nsIRunnable)
|
|
NS_INTERFACE_MAP_ENTRY_AMBIGUOUS(nsISupports, nsIContentPermissionRequest)
|
|
NS_INTERFACE_MAP_END
|
|
|
|
NS_IMPL_CYCLE_COLLECTING_ADDREF(CameraPermissionRequest)
|
|
NS_IMPL_CYCLE_COLLECTING_RELEASE(CameraPermissionRequest)
|
|
|
|
NS_IMETHODIMP
|
|
CameraPermissionRequest::Run()
|
|
{
|
|
return nsContentPermissionUtils::AskPermission(this, mWindow);
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
CameraPermissionRequest::GetPrincipal(nsIPrincipal** aRequestingPrincipal)
|
|
{
|
|
NS_ADDREF(*aRequestingPrincipal = mPrincipal);
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
CameraPermissionRequest::GetWindow(nsIDOMWindow** aRequestingWindow)
|
|
{
|
|
NS_ADDREF(*aRequestingWindow = mWindow);
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
CameraPermissionRequest::GetElement(nsIDOMElement** aElement)
|
|
{
|
|
*aElement = nullptr;
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
CameraPermissionRequest::Cancel()
|
|
{
|
|
return DispatchCallback(nsIPermissionManager::DENY_ACTION);
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
CameraPermissionRequest::Allow(JS::HandleValue aChoices)
|
|
{
|
|
MOZ_ASSERT(aChoices.isUndefined());
|
|
return DispatchCallback(nsIPermissionManager::ALLOW_ACTION);
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
CameraPermissionRequest::GetRequester(nsIContentPermissionRequester** aRequester)
|
|
{
|
|
NS_ENSURE_ARG_POINTER(aRequester);
|
|
|
|
nsCOMPtr<nsIContentPermissionRequester> requester = mRequester;
|
|
requester.forget(aRequester);
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
CameraPermissionRequest::DispatchCallback(uint32_t aPermission)
|
|
{
|
|
nsCOMPtr<nsIRunnable> callbackRunnable;
|
|
if (aPermission == nsIPermissionManager::ALLOW_ACTION) {
|
|
callbackRunnable = NS_NewRunnableMethod(this, &CameraPermissionRequest::CallAllow);
|
|
} else {
|
|
callbackRunnable = NS_NewRunnableMethod(this, &CameraPermissionRequest::CallCancel);
|
|
}
|
|
return NS_DispatchToMainThread(callbackRunnable);
|
|
}
|
|
|
|
void
|
|
CameraPermissionRequest::CallAllow()
|
|
{
|
|
mCameraManager->PermissionAllowed(mCameraId, mInitialConfig, mPromise);
|
|
}
|
|
|
|
void
|
|
CameraPermissionRequest::CallCancel()
|
|
{
|
|
mCameraManager->PermissionCancelled(mCameraId, mInitialConfig, mPromise);
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
CameraPermissionRequest::GetTypes(nsIArray** aTypes)
|
|
{
|
|
nsTArray<nsString> emptyOptions;
|
|
return nsContentPermissionUtils::CreatePermissionArray(NS_LITERAL_CSTRING("camera"),
|
|
NS_LITERAL_CSTRING("unused"),
|
|
emptyOptions,
|
|
aTypes);
|
|
}
|
|
|
|
#ifdef MOZ_WIDGET_GONK
|
|
/* static */ void
|
|
nsDOMCameraManager::PreinitCameraHardware()
|
|
{
|
|
nsDOMCameraControl::PreinitCameraHardware();
|
|
}
|
|
#endif
|
|
|
|
already_AddRefed<Promise>
|
|
nsDOMCameraManager::GetCamera(const nsAString& aCamera,
|
|
const CameraConfiguration& aInitialConfig,
|
|
ErrorResult& aRv)
|
|
{
|
|
DOM_CAMERA_LOGT("%s:%d\n", __func__, __LINE__);
|
|
|
|
uint32_t cameraId = 0; // back (or forward-facing) camera by default
|
|
if (aCamera.EqualsLiteral("front")) {
|
|
cameraId = 1;
|
|
}
|
|
|
|
nsCOMPtr<nsIGlobalObject> global = do_QueryInterface(mWindow);
|
|
if (!global) {
|
|
aRv.Throw(NS_ERROR_FAILURE);
|
|
return nullptr;
|
|
}
|
|
|
|
nsRefPtr<Promise> promise = Promise::Create(global, aRv);
|
|
if (aRv.Failed()) {
|
|
return nullptr;
|
|
}
|
|
|
|
if (mPermission == nsIPermissionManager::ALLOW_ACTION) {
|
|
PermissionAllowed(cameraId, aInitialConfig, promise);
|
|
return promise.forget();
|
|
}
|
|
|
|
nsCOMPtr<nsIScriptObjectPrincipal> sop = do_QueryInterface(mWindow);
|
|
if (!sop) {
|
|
aRv.Throw(NS_ERROR_UNEXPECTED);
|
|
return nullptr;
|
|
}
|
|
|
|
nsCOMPtr<nsIPrincipal> principal = sop->GetPrincipal();
|
|
// If we are a CERTIFIED app, we can short-circuit the permission check,
|
|
// which gets us a performance win.
|
|
uint16_t status = nsIPrincipal::APP_STATUS_NOT_INSTALLED;
|
|
principal->GetAppStatus(&status);
|
|
// Unprivileged mochitests always fail the dispatched permission check,
|
|
// even if permission to the camera has been granted.
|
|
bool immediateCheck = false;
|
|
CameraPreferences::GetPref("camera.control.test.permission", immediateCheck);
|
|
if ((status == nsIPrincipal::APP_STATUS_CERTIFIED || immediateCheck) && CheckPermission(mWindow)) {
|
|
PermissionAllowed(cameraId, aInitialConfig, promise);
|
|
return promise.forget();
|
|
}
|
|
|
|
nsCOMPtr<nsIRunnable> permissionRequest =
|
|
new CameraPermissionRequest(principal, mWindow, this, cameraId,
|
|
aInitialConfig, promise);
|
|
|
|
NS_DispatchToMainThread(permissionRequest);
|
|
return promise.forget();
|
|
}
|
|
|
|
void
|
|
nsDOMCameraManager::PermissionAllowed(uint32_t aCameraId,
|
|
const CameraConfiguration& aInitialConfig,
|
|
Promise* aPromise)
|
|
{
|
|
mPermission = nsIPermissionManager::ALLOW_ACTION;
|
|
|
|
// Creating this object will trigger the aOnSuccess callback
|
|
// (or the aOnError one, if it fails).
|
|
nsRefPtr<nsDOMCameraControl> cameraControl =
|
|
new nsDOMCameraControl(aCameraId, aInitialConfig, aPromise, mWindow);
|
|
|
|
Register(cameraControl);
|
|
}
|
|
|
|
void
|
|
nsDOMCameraManager::PermissionCancelled(uint32_t aCameraId,
|
|
const CameraConfiguration& aInitialConfig,
|
|
Promise* aPromise)
|
|
{
|
|
mPermission = nsIPermissionManager::DENY_ACTION;
|
|
aPromise->MaybeReject(NS_ERROR_DOM_SECURITY_ERR);
|
|
}
|
|
|
|
void
|
|
nsDOMCameraManager::Register(nsDOMCameraControl* aDOMCameraControl)
|
|
{
|
|
DOM_CAMERA_LOGI(">>> Register( aDOMCameraControl = %p ) mWindowId = 0x%" PRIx64 "\n", aDOMCameraControl, mWindowId);
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
CameraControls* controls = sActiveWindows->Get(mWindowId);
|
|
if (!controls) {
|
|
controls = new CameraControls();
|
|
sActiveWindows->Put(mWindowId, controls);
|
|
}
|
|
|
|
// Remove any stale CameraControl objects to limit our memory usage
|
|
uint32_t i = controls->Length();
|
|
while (i > 0) {
|
|
--i;
|
|
nsRefPtr<nsDOMCameraControl> cameraControl =
|
|
do_QueryObject(controls->ElementAt(i));
|
|
if (!cameraControl) {
|
|
controls->RemoveElementAt(i);
|
|
}
|
|
}
|
|
|
|
// Put the camera control into the hash table
|
|
nsWeakPtr cameraControl =
|
|
do_GetWeakReference(static_cast<DOMMediaStream*>(aDOMCameraControl));
|
|
controls->AppendElement(cameraControl);
|
|
}
|
|
|
|
void
|
|
nsDOMCameraManager::Shutdown(uint64_t aWindowId)
|
|
{
|
|
DOM_CAMERA_LOGI(">>> Shutdown( aWindowId = 0x%" PRIx64 " )\n", aWindowId);
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
CameraControls* controls = sActiveWindows->Get(aWindowId);
|
|
if (!controls) {
|
|
return;
|
|
}
|
|
|
|
uint32_t i = controls->Length();
|
|
while (i > 0) {
|
|
--i;
|
|
nsRefPtr<nsDOMCameraControl> cameraControl =
|
|
do_QueryObject(controls->ElementAt(i));
|
|
if (cameraControl) {
|
|
cameraControl->Shutdown();
|
|
}
|
|
}
|
|
controls->Clear();
|
|
|
|
sActiveWindows->Remove(aWindowId);
|
|
}
|
|
|
|
void
|
|
nsDOMCameraManager::XpComShutdown()
|
|
{
|
|
DOM_CAMERA_LOGI(">>> XPCOM Shutdown\n");
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
nsCOMPtr<nsIObserverService> obs = services::GetObserverService();
|
|
obs->RemoveObserver(this, "xpcom-shutdown");
|
|
|
|
delete sActiveWindows;
|
|
sActiveWindows = nullptr;
|
|
}
|
|
|
|
nsresult
|
|
nsDOMCameraManager::Observe(nsISupports* aSubject, const char* aTopic, const char16_t* aData)
|
|
{
|
|
if (strcmp(aTopic, "xpcom-shutdown") == 0) {
|
|
XpComShutdown();
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
void
|
|
nsDOMCameraManager::OnNavigation(uint64_t aWindowId)
|
|
{
|
|
DOM_CAMERA_LOGI(">>> OnNavigation event\n");
|
|
Shutdown(aWindowId);
|
|
}
|
|
|
|
bool
|
|
nsDOMCameraManager::IsWindowStillActive(uint64_t aWindowId)
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
if (!sActiveWindows) {
|
|
return false;
|
|
}
|
|
|
|
return !!sActiveWindows->Get(aWindowId);
|
|
}
|
|
|
|
JSObject*
|
|
nsDOMCameraManager::WrapObject(JSContext* aCx, JS::Handle<JSObject*> aGivenProto)
|
|
{
|
|
return CameraManagerBinding::Wrap(aCx, this, aGivenProto);
|
|
}
|