mirror of
https://github.com/roytam1/palemoon27.git
synced 2026-05-26 14:30:27 +00:00
9e41bd2207
- Bug 1230352 - Update to Oculus SDK 0.8.0,r=vlad (ec5404763a) - Bug 1237689 - Enable Oculus hardware latency tester r=daoshengmu (a914fb5c78) - Bug 1248757 Use string ::Assign() instead of Adopt() when reading blobs as strings. r=asuth (65e361797f) - Bug 1248757 followup - Release blob after assignment in DoGetBlobAsString on CLOSED TREE. (8f5d86a27a) - Bug 1240583 - Odin: record and assert whether a function is defined yet (r=bbouvier) (693c0be6fe) - Bug 1240583 - Odin: store code-range index instead of entry offset in ModuleGenerator (r=bbouvier) (ac14028824) - Bug 1240583 - add thunkWithPatch/patchThunk (r=jandem) (9b53ac4d5b) - Bug 1240583 - Odin: fix long jumps/calls on ARM for large modules (r=bbouvier) (3f7bacc9cf) - Bug 1240583 - Odin: add MacroAssembler::repatchThunk (r=bbouvier) (1336b1492d) - Bug 1037483 adopt microformats-shiv for microformats v2 support, r=tantek (c08afa618d) - Bug 1224766 - forward callID to disambiguate multiple gUM requests from same window. r=jesup,smaug (547eafdbaa) - Bug 1201393. Remove irrelevant ProcessedMediaStream for nsSpeechTask. r=eitan (f17f8e6821) - Bug 1240583: Fix non-unified build for fuzzers; r=luke (cc7ea34899) - Bug 1212366 - Part 1. Don't call SetAudioOutputVolume if stream is destroyed. r=roc (af1106491c) - Bug 1212366 - Part 2. Don't release SourceMediaStream until StreamListener is called. r=roc (08cf9cf62f) - Bug 1220320 - implement the nsSupportsWeakReference. r=baku (0cfd32c83a) - Bug 1225347 - Apply audio setting to volume parameter of Speak(). r=eeejay (fc6dbd938c) - Bug 1228564 - part 1 : revert the changeset of bug 1190040. r=baku. (921a0f7383) - Bug 1195051 - Part 3: Test changes; r=padenot (438a73a408) - Bug 1195051 - Part 4: Fix a null pointer crash happening after the destination node gets CCed (90fd50c8ac) - Bug 1228564 - part 2 : check audio capturing when the agent is registered/unregistered. r=baku. (14f473625b) - Bug 1228564 - Follow-up to fix static analysis build bustage. r=me (295d61a1b6) - Bug 1247846 - Odin: switch CallIndirect to wasm binary encoding (r=bbouvier) (5997b4c59b) - Bug 1247846 - Odin: refactor ModuleGenerator::finish (r=bbouvier) (a71293c284) - Bug 1247846 - Odin: refactor error stub generation (r=bbouvier) (5f9f98b215)
325 lines
9.2 KiB
C++
325 lines
9.2 KiB
C++
/* -*- 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 "AudioChannelAgent.h"
|
|
#include "AudioChannelService.h"
|
|
#include "mozilla/Preferences.h"
|
|
#include "nsIAppsService.h"
|
|
#include "nsIDocument.h"
|
|
#include "nsIDOMWindow.h"
|
|
#include "nsIPrincipal.h"
|
|
#include "nsPIDOMWindow.h"
|
|
#include "nsXULAppAPI.h"
|
|
|
|
using namespace mozilla::dom;
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_CLASS(AudioChannelAgent)
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK_BEGIN(AudioChannelAgent)
|
|
tmp->Shutdown();
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK(mWindow)
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK(mCallback)
|
|
NS_IMPL_CYCLE_COLLECTION_UNLINK_END
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN(AudioChannelAgent)
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mWindow)
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mCallback)
|
|
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END
|
|
|
|
NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION(AudioChannelAgent)
|
|
NS_INTERFACE_MAP_ENTRY(nsIAudioChannelAgent)
|
|
NS_INTERFACE_MAP_ENTRY(nsISupports)
|
|
NS_INTERFACE_MAP_END
|
|
|
|
NS_IMPL_CYCLE_COLLECTING_ADDREF(AudioChannelAgent)
|
|
NS_IMPL_CYCLE_COLLECTING_RELEASE(AudioChannelAgent)
|
|
|
|
AudioChannelAgent::AudioChannelAgent()
|
|
: mAudioChannelType(AUDIO_AGENT_CHANNEL_ERROR)
|
|
, mInnerWindowID(0)
|
|
, mIsRegToService(false)
|
|
{
|
|
}
|
|
|
|
AudioChannelAgent::~AudioChannelAgent()
|
|
{
|
|
Shutdown();
|
|
}
|
|
|
|
void
|
|
AudioChannelAgent::Shutdown()
|
|
{
|
|
if (mIsRegToService) {
|
|
NotifyStoppedPlaying();
|
|
}
|
|
}
|
|
|
|
NS_IMETHODIMP AudioChannelAgent::GetAudioChannelType(int32_t *aAudioChannelType)
|
|
{
|
|
*aAudioChannelType = mAudioChannelType;
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
AudioChannelAgent::Init(nsIDOMWindow* aWindow, int32_t aChannelType,
|
|
nsIAudioChannelAgentCallback *aCallback)
|
|
{
|
|
return InitInternal(aWindow, aChannelType, aCallback,
|
|
/* useWeakRef = */ false);
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
AudioChannelAgent::InitWithWeakCallback(nsIDOMWindow* aWindow,
|
|
int32_t aChannelType,
|
|
nsIAudioChannelAgentCallback *aCallback)
|
|
{
|
|
return InitInternal(aWindow, aChannelType, aCallback,
|
|
/* useWeakRef = */ true);
|
|
}
|
|
|
|
nsresult
|
|
AudioChannelAgent::FindCorrectWindow(nsIDOMWindow* aWindow)
|
|
{
|
|
nsCOMPtr<nsPIDOMWindow> window = do_QueryInterface(aWindow);
|
|
MOZ_ASSERT(window->IsInnerWindow());
|
|
|
|
mWindow = window->GetScriptableTop();
|
|
if (NS_WARN_IF(!mWindow)) {
|
|
return NS_OK;
|
|
}
|
|
|
|
mWindow = mWindow->GetOuterWindow();
|
|
if (NS_WARN_IF(!mWindow)) {
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
// From here we do an hack for nested iframes.
|
|
// The system app doesn't have access to the nested iframe objects so it
|
|
// cannot control the volume of the agents running in nested apps. What we do
|
|
// here is to assign those Agents to the top scriptable window of the parent
|
|
// iframe (what is controlled by the system app).
|
|
// For doing this we go recursively back into the chain of windows until we
|
|
// find apps that are not the system one.
|
|
window = mWindow->GetParent();
|
|
if (!window || window == mWindow) {
|
|
return NS_OK;
|
|
}
|
|
|
|
window = window->GetCurrentInnerWindow();
|
|
if (!window) {
|
|
return NS_OK;
|
|
}
|
|
|
|
nsCOMPtr<nsIDocument> doc = window->GetExtantDoc();
|
|
if (!doc) {
|
|
return NS_OK;
|
|
}
|
|
|
|
nsCOMPtr<nsIPrincipal> principal = doc->NodePrincipal();
|
|
|
|
uint32_t appId;
|
|
nsresult rv = principal->GetAppId(&appId);
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return rv;
|
|
}
|
|
|
|
if (appId == nsIScriptSecurityManager::NO_APP_ID ||
|
|
appId == nsIScriptSecurityManager::UNKNOWN_APP_ID) {
|
|
return NS_OK;
|
|
}
|
|
|
|
nsCOMPtr<nsIAppsService> appsService = do_GetService(APPS_SERVICE_CONTRACTID);
|
|
if (NS_WARN_IF(!appsService)) {
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
nsAdoptingString systemAppManifest =
|
|
mozilla::Preferences::GetString("b2g.system_manifest_url");
|
|
if (!systemAppManifest) {
|
|
return NS_OK;
|
|
}
|
|
|
|
uint32_t systemAppId;
|
|
rv = appsService->GetAppLocalIdByManifestURL(systemAppManifest, &systemAppId);
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return rv;
|
|
}
|
|
|
|
if (systemAppId == appId) {
|
|
return NS_OK;
|
|
}
|
|
|
|
return FindCorrectWindow(window);
|
|
}
|
|
|
|
nsresult
|
|
AudioChannelAgent::InitInternal(nsIDOMWindow* aWindow, int32_t aChannelType,
|
|
nsIAudioChannelAgentCallback *aCallback,
|
|
bool aUseWeakRef)
|
|
{
|
|
// We syncd the enum of channel type between nsIAudioChannelAgent.idl and
|
|
// AudioChannelBinding.h the same.
|
|
MOZ_ASSERT(int(AUDIO_AGENT_CHANNEL_NORMAL) == int(AudioChannel::Normal) &&
|
|
int(AUDIO_AGENT_CHANNEL_CONTENT) == int(AudioChannel::Content) &&
|
|
int(AUDIO_AGENT_CHANNEL_NOTIFICATION) == int(AudioChannel::Notification) &&
|
|
int(AUDIO_AGENT_CHANNEL_ALARM) == int(AudioChannel::Alarm) &&
|
|
int(AUDIO_AGENT_CHANNEL_TELEPHONY) == int(AudioChannel::Telephony) &&
|
|
int(AUDIO_AGENT_CHANNEL_RINGER) == int(AudioChannel::Ringer) &&
|
|
int(AUDIO_AGENT_CHANNEL_SYSTEM) == int(AudioChannel::System) &&
|
|
int(AUDIO_AGENT_CHANNEL_PUBLICNOTIFICATION) == int(AudioChannel::Publicnotification),
|
|
"Enum of channel on nsIAudioChannelAgent.idl should be the same with AudioChannelBinding.h");
|
|
|
|
if (mAudioChannelType != AUDIO_AGENT_CHANNEL_ERROR ||
|
|
aChannelType > AUDIO_AGENT_CHANNEL_SYSTEM ||
|
|
aChannelType < AUDIO_AGENT_CHANNEL_NORMAL) {
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
if (NS_WARN_IF(!aWindow)) {
|
|
return NS_OK;
|
|
}
|
|
|
|
nsCOMPtr<nsPIDOMWindow> pInnerWindow = do_QueryInterface(aWindow);
|
|
MOZ_ASSERT(pInnerWindow->IsInnerWindow());
|
|
mInnerWindowID = pInnerWindow->WindowID();
|
|
|
|
nsresult rv = FindCorrectWindow(aWindow);
|
|
if (NS_WARN_IF(NS_FAILED(rv))) {
|
|
return rv;
|
|
}
|
|
|
|
mAudioChannelType = aChannelType;
|
|
|
|
if (aUseWeakRef) {
|
|
mWeakCallback = do_GetWeakReference(aCallback);
|
|
} else {
|
|
mCallback = aCallback;
|
|
}
|
|
|
|
MOZ_LOG(AudioChannelService::GetAudioChannelLog(), LogLevel::Debug,
|
|
("AudioChannelAgent, InitInternal, this = %p, type = %d, "
|
|
"owner = %p, hasCallback = %d\n", this, mAudioChannelType,
|
|
mWindow.get(), (!!mCallback || !!mWeakCallback)));
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP AudioChannelAgent::NotifyStartedPlaying(float *aVolume,
|
|
bool* aMuted)
|
|
{
|
|
MOZ_ASSERT(aVolume);
|
|
MOZ_ASSERT(aMuted);
|
|
|
|
// Window-less AudioChannelAgents are muted by default.
|
|
if (!mWindow) {
|
|
*aVolume = 0;
|
|
*aMuted = true;
|
|
return NS_OK;
|
|
}
|
|
|
|
RefPtr<AudioChannelService> service = AudioChannelService::GetOrCreate();
|
|
if (mAudioChannelType == AUDIO_AGENT_CHANNEL_ERROR ||
|
|
service == nullptr || mIsRegToService) {
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
service->RegisterAudioChannelAgent(this,
|
|
static_cast<AudioChannel>(mAudioChannelType));
|
|
|
|
service->GetState(mWindow, mAudioChannelType, aVolume, aMuted);
|
|
|
|
MOZ_LOG(AudioChannelService::GetAudioChannelLog(), LogLevel::Debug,
|
|
("AudioChannelAgent, NotifyStartedPlaying, this = %p, mute = %d, "
|
|
"volume = %f\n", this, *aMuted, *aVolume));
|
|
|
|
mIsRegToService = true;
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP AudioChannelAgent::NotifyStoppedPlaying()
|
|
{
|
|
if (mAudioChannelType == AUDIO_AGENT_CHANNEL_ERROR ||
|
|
!mIsRegToService) {
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
|
|
MOZ_LOG(AudioChannelService::GetAudioChannelLog(), LogLevel::Debug,
|
|
("AudioChannelAgent, NotifyStoppedPlaying, this = %p\n", this));
|
|
|
|
RefPtr<AudioChannelService> service = AudioChannelService::GetOrCreate();
|
|
if (service) {
|
|
service->UnregisterAudioChannelAgent(this);
|
|
}
|
|
|
|
mIsRegToService = false;
|
|
return NS_OK;
|
|
}
|
|
|
|
already_AddRefed<nsIAudioChannelAgentCallback>
|
|
AudioChannelAgent::GetCallback()
|
|
{
|
|
nsCOMPtr<nsIAudioChannelAgentCallback> callback = mCallback;
|
|
if (!callback) {
|
|
callback = do_QueryReferent(mWeakCallback);
|
|
}
|
|
return callback.forget();
|
|
}
|
|
|
|
void
|
|
AudioChannelAgent::WindowVolumeChanged()
|
|
{
|
|
nsCOMPtr<nsIAudioChannelAgentCallback> callback = GetCallback();
|
|
if (!callback) {
|
|
return;
|
|
}
|
|
|
|
float volume = 1.0;
|
|
bool muted = false;
|
|
|
|
RefPtr<AudioChannelService> service = AudioChannelService::GetOrCreate();
|
|
if (service) {
|
|
service->GetState(mWindow, mAudioChannelType, &volume, &muted);
|
|
}
|
|
|
|
MOZ_LOG(AudioChannelService::GetAudioChannelLog(), LogLevel::Debug,
|
|
("AudioChannelAgent, WindowVolumeChanged, this = %p, mute = %d, "
|
|
"volume = %f\n", this, muted, volume));
|
|
|
|
callback->WindowVolumeChanged(volume, muted);
|
|
}
|
|
|
|
uint64_t
|
|
AudioChannelAgent::WindowID() const
|
|
{
|
|
return mWindow ? mWindow->WindowID() : 0;
|
|
}
|
|
|
|
uint64_t
|
|
AudioChannelAgent::InnerWindowID() const
|
|
{
|
|
return mInnerWindowID;
|
|
}
|
|
|
|
void
|
|
AudioChannelAgent::WindowAudioCaptureChanged(uint64_t aInnerWindowID,
|
|
bool aCapture)
|
|
{
|
|
if (aInnerWindowID != mInnerWindowID) {
|
|
return;
|
|
}
|
|
|
|
nsCOMPtr<nsIAudioChannelAgentCallback> callback = GetCallback();
|
|
if (!callback) {
|
|
return;
|
|
}
|
|
|
|
MOZ_LOG(AudioChannelService::GetAudioChannelLog(), LogLevel::Debug,
|
|
("AudioChannelAgent, WindowAudioCaptureChanged, this = %p, "
|
|
"capture = %d\n", this, aCapture));
|
|
|
|
callback->WindowAudioCaptureChanged(aCapture);
|
|
}
|