mirror of
https://github.com/roytam1/palemoon27.git
synced 2026-05-26 14:18:48 +00:00
7bd3227e5d
- Bug 1141282 - DynamicCompressorNode's readonly 'reduction' should be a float. r=ehsan (4685d2cf8) - Bug 1153783 - Implement the `detune` AudioParam for the AudioBufferSourceNode. r=ehsan (8e6b3aca4) - Bug 1134034 - Add a chrome-only parentId and name on AudioParam for devtools. r=ehsan (13815cb94) - Bug 1159290 - "Add final/override to WebAudio classes". r=padenot (83cb303ea) - Bug 1161946 - MainThreadMediaStreamListener should be notified just when the stream is finished - patch 1, r=padenot (c2a126103) - Bug 1161946 - MainThreadMediaStreamListener should be notified just when the stream is finished - patch 2, r=padenot (a1f408cce) - Bug 1161946 - patch 3 - explicit CTOR for NotifyRunnable CLOSED TREE (f304c1ef0) - Bug 974089 - Destroy WebAudio MediaStream when a source finishes. r=padenot (66d7e20df) - Bug 1175510 - Update the AudioBufferSourceNode <=> PannerNode mapping when destroying AudioNodeStream. r=karlt (156741d15) - bug 1179662 uninline DestroyMediaStream overrides so that mStream type need not be complete r=padenot (08d0ebd62) - bug 1179662 specify AudioNode::mStream as AudioNodeStream r=padenot (3aaa3a544) - bug 1179662 handle null AudioNodeStream stream r=padenot (980a8296b) - bug 914392 don't check engine's Node existence in ProcessBlock r=padenot (583176721)
275 lines
8.9 KiB
C++
275 lines
8.9 KiB
C++
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/* vim:set ts=2 sw=2 sts=2 et cindent: */
|
|
/* 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 "DynamicsCompressorNode.h"
|
|
#include "mozilla/dom/DynamicsCompressorNodeBinding.h"
|
|
#include "AudioNodeEngine.h"
|
|
#include "AudioNodeStream.h"
|
|
#include "AudioDestinationNode.h"
|
|
#include "WebAudioUtils.h"
|
|
#include "blink/DynamicsCompressor.h"
|
|
|
|
using WebCore::DynamicsCompressor;
|
|
|
|
namespace mozilla {
|
|
namespace dom {
|
|
|
|
NS_IMPL_CYCLE_COLLECTION_INHERITED(DynamicsCompressorNode, AudioNode,
|
|
mThreshold,
|
|
mKnee,
|
|
mRatio,
|
|
mAttack,
|
|
mRelease)
|
|
|
|
NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION_INHERITED(DynamicsCompressorNode)
|
|
NS_INTERFACE_MAP_END_INHERITING(AudioNode)
|
|
|
|
NS_IMPL_ADDREF_INHERITED(DynamicsCompressorNode, AudioNode)
|
|
NS_IMPL_RELEASE_INHERITED(DynamicsCompressorNode, AudioNode)
|
|
|
|
class DynamicsCompressorNodeEngine final : public AudioNodeEngine
|
|
{
|
|
public:
|
|
explicit DynamicsCompressorNodeEngine(AudioNode* aNode,
|
|
AudioDestinationNode* aDestination)
|
|
: AudioNodeEngine(aNode)
|
|
, mSource(nullptr)
|
|
, mDestination(aDestination->Stream())
|
|
// Keep the default value in sync with the default value in
|
|
// DynamicsCompressorNode::DynamicsCompressorNode.
|
|
, mThreshold(-24.f)
|
|
, mKnee(30.f)
|
|
, mRatio(12.f)
|
|
, mAttack(0.003f)
|
|
, mRelease(0.25f)
|
|
, mCompressor(new DynamicsCompressor(mDestination->SampleRate(), 2))
|
|
{
|
|
}
|
|
|
|
void SetSourceStream(AudioNodeStream* aSource)
|
|
{
|
|
mSource = aSource;
|
|
}
|
|
|
|
enum Parameters {
|
|
THRESHOLD,
|
|
KNEE,
|
|
RATIO,
|
|
ATTACK,
|
|
RELEASE
|
|
};
|
|
void SetTimelineParameter(uint32_t aIndex,
|
|
const AudioParamTimeline& aValue,
|
|
TrackRate aSampleRate) override
|
|
{
|
|
MOZ_ASSERT(mSource && mDestination);
|
|
switch (aIndex) {
|
|
case THRESHOLD:
|
|
mThreshold = aValue;
|
|
WebAudioUtils::ConvertAudioParamToTicks(mThreshold, mSource, mDestination);
|
|
break;
|
|
case KNEE:
|
|
mKnee = aValue;
|
|
WebAudioUtils::ConvertAudioParamToTicks(mKnee, mSource, mDestination);
|
|
break;
|
|
case RATIO:
|
|
mRatio = aValue;
|
|
WebAudioUtils::ConvertAudioParamToTicks(mRatio, mSource, mDestination);
|
|
break;
|
|
case ATTACK:
|
|
mAttack = aValue;
|
|
WebAudioUtils::ConvertAudioParamToTicks(mAttack, mSource, mDestination);
|
|
break;
|
|
case RELEASE:
|
|
mRelease = aValue;
|
|
WebAudioUtils::ConvertAudioParamToTicks(mRelease, mSource, mDestination);
|
|
break;
|
|
default:
|
|
NS_ERROR("Bad DynamicsCompresssorNodeEngine TimelineParameter");
|
|
}
|
|
}
|
|
|
|
virtual void ProcessBlock(AudioNodeStream* aStream,
|
|
const AudioChunk& aInput,
|
|
AudioChunk* aOutput,
|
|
bool* aFinished) override
|
|
{
|
|
if (aInput.IsNull()) {
|
|
// Just output silence
|
|
*aOutput = aInput;
|
|
return;
|
|
}
|
|
|
|
const uint32_t channelCount = aInput.mChannelData.Length();
|
|
if (mCompressor->numberOfChannels() != channelCount) {
|
|
// Create a new compressor object with a new channel count
|
|
mCompressor = new WebCore::DynamicsCompressor(aStream->SampleRate(),
|
|
aInput.mChannelData.Length());
|
|
}
|
|
|
|
StreamTime pos = aStream->GetCurrentPosition();
|
|
mCompressor->setParameterValue(DynamicsCompressor::ParamThreshold,
|
|
mThreshold.GetValueAtTime(pos));
|
|
mCompressor->setParameterValue(DynamicsCompressor::ParamKnee,
|
|
mKnee.GetValueAtTime(pos));
|
|
mCompressor->setParameterValue(DynamicsCompressor::ParamRatio,
|
|
mRatio.GetValueAtTime(pos));
|
|
mCompressor->setParameterValue(DynamicsCompressor::ParamAttack,
|
|
mAttack.GetValueAtTime(pos));
|
|
mCompressor->setParameterValue(DynamicsCompressor::ParamRelease,
|
|
mRelease.GetValueAtTime(pos));
|
|
|
|
AllocateAudioBlock(channelCount, aOutput);
|
|
mCompressor->process(&aInput, aOutput, aInput.GetDuration());
|
|
|
|
SendReductionParamToMainThread(aStream,
|
|
mCompressor->parameterValue(DynamicsCompressor::ParamReduction));
|
|
}
|
|
|
|
virtual size_t SizeOfExcludingThis(MallocSizeOf aMallocSizeOf) const override
|
|
{
|
|
// Not owned:
|
|
// - mSource (probably)
|
|
// - mDestination (probably)
|
|
// - Don't count the AudioParamTimelines, their inner refs are owned by the
|
|
// AudioNode.
|
|
size_t amount = AudioNodeEngine::SizeOfExcludingThis(aMallocSizeOf);
|
|
amount += mCompressor->sizeOfIncludingThis(aMallocSizeOf);
|
|
return amount;
|
|
}
|
|
|
|
virtual size_t SizeOfIncludingThis(MallocSizeOf aMallocSizeOf) const override
|
|
{
|
|
return aMallocSizeOf(this) + SizeOfExcludingThis(aMallocSizeOf);
|
|
}
|
|
|
|
private:
|
|
void SendReductionParamToMainThread(AudioNodeStream* aStream, float aReduction)
|
|
{
|
|
MOZ_ASSERT(!NS_IsMainThread());
|
|
|
|
class Command final : public nsRunnable
|
|
{
|
|
public:
|
|
Command(AudioNodeStream* aStream, float aReduction)
|
|
: mStream(aStream)
|
|
, mReduction(aReduction)
|
|
{
|
|
}
|
|
|
|
NS_IMETHOD Run() override
|
|
{
|
|
nsRefPtr<DynamicsCompressorNode> node =
|
|
static_cast<DynamicsCompressorNode*>
|
|
(mStream->Engine()->NodeMainThread());
|
|
if (node) {
|
|
node->SetReduction(mReduction);
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
private:
|
|
nsRefPtr<AudioNodeStream> mStream;
|
|
float mReduction;
|
|
};
|
|
|
|
NS_DispatchToMainThread(new Command(aStream, aReduction));
|
|
}
|
|
|
|
private:
|
|
AudioNodeStream* mSource;
|
|
AudioNodeStream* mDestination;
|
|
AudioParamTimeline mThreshold;
|
|
AudioParamTimeline mKnee;
|
|
AudioParamTimeline mRatio;
|
|
AudioParamTimeline mAttack;
|
|
AudioParamTimeline mRelease;
|
|
nsAutoPtr<DynamicsCompressor> mCompressor;
|
|
};
|
|
|
|
DynamicsCompressorNode::DynamicsCompressorNode(AudioContext* aContext)
|
|
: AudioNode(aContext,
|
|
2,
|
|
ChannelCountMode::Explicit,
|
|
ChannelInterpretation::Speakers)
|
|
, mThreshold(new AudioParam(this, SendThresholdToStream, -24.f, "threshold"))
|
|
, mKnee(new AudioParam(this, SendKneeToStream, 30.f, "knee"))
|
|
, mRatio(new AudioParam(this, SendRatioToStream, 12.f, "ratio"))
|
|
, mReduction(0)
|
|
, mAttack(new AudioParam(this, SendAttackToStream, 0.003f, "attack"))
|
|
, mRelease(new AudioParam(this, SendReleaseToStream, 0.25f, "release"))
|
|
{
|
|
DynamicsCompressorNodeEngine* engine = new DynamicsCompressorNodeEngine(this, aContext->Destination());
|
|
mStream = aContext->Graph()->CreateAudioNodeStream(engine, MediaStreamGraph::INTERNAL_STREAM);
|
|
engine->SetSourceStream(mStream);
|
|
}
|
|
|
|
DynamicsCompressorNode::~DynamicsCompressorNode()
|
|
{
|
|
}
|
|
|
|
size_t
|
|
DynamicsCompressorNode::SizeOfExcludingThis(MallocSizeOf aMallocSizeOf) const
|
|
{
|
|
size_t amount = AudioNode::SizeOfExcludingThis(aMallocSizeOf);
|
|
amount += mThreshold->SizeOfIncludingThis(aMallocSizeOf);
|
|
amount += mKnee->SizeOfIncludingThis(aMallocSizeOf);
|
|
amount += mRatio->SizeOfIncludingThis(aMallocSizeOf);
|
|
amount += mAttack->SizeOfIncludingThis(aMallocSizeOf);
|
|
amount += mRelease->SizeOfIncludingThis(aMallocSizeOf);
|
|
return amount;
|
|
}
|
|
|
|
size_t
|
|
DynamicsCompressorNode::SizeOfIncludingThis(MallocSizeOf aMallocSizeOf) const
|
|
{
|
|
return aMallocSizeOf(this) + SizeOfExcludingThis(aMallocSizeOf);
|
|
}
|
|
|
|
JSObject*
|
|
DynamicsCompressorNode::WrapObject(JSContext* aCx, JS::Handle<JSObject*> aGivenProto)
|
|
{
|
|
return DynamicsCompressorNodeBinding::Wrap(aCx, this, aGivenProto);
|
|
}
|
|
|
|
void
|
|
DynamicsCompressorNode::SendThresholdToStream(AudioNode* aNode)
|
|
{
|
|
DynamicsCompressorNode* This = static_cast<DynamicsCompressorNode*>(aNode);
|
|
SendTimelineParameterToStream(This, DynamicsCompressorNodeEngine::THRESHOLD, *This->mThreshold);
|
|
}
|
|
|
|
void
|
|
DynamicsCompressorNode::SendKneeToStream(AudioNode* aNode)
|
|
{
|
|
DynamicsCompressorNode* This = static_cast<DynamicsCompressorNode*>(aNode);
|
|
SendTimelineParameterToStream(This, DynamicsCompressorNodeEngine::KNEE, *This->mKnee);
|
|
}
|
|
|
|
void
|
|
DynamicsCompressorNode::SendRatioToStream(AudioNode* aNode)
|
|
{
|
|
DynamicsCompressorNode* This = static_cast<DynamicsCompressorNode*>(aNode);
|
|
SendTimelineParameterToStream(This, DynamicsCompressorNodeEngine::RATIO, *This->mRatio);
|
|
}
|
|
|
|
void
|
|
DynamicsCompressorNode::SendAttackToStream(AudioNode* aNode)
|
|
{
|
|
DynamicsCompressorNode* This = static_cast<DynamicsCompressorNode*>(aNode);
|
|
SendTimelineParameterToStream(This, DynamicsCompressorNodeEngine::ATTACK, *This->mAttack);
|
|
}
|
|
|
|
void
|
|
DynamicsCompressorNode::SendReleaseToStream(AudioNode* aNode)
|
|
{
|
|
DynamicsCompressorNode* This = static_cast<DynamicsCompressorNode*>(aNode);
|
|
SendTimelineParameterToStream(This, DynamicsCompressorNodeEngine::RELEASE, *This->mRelease);
|
|
}
|
|
|
|
} // namespace dom
|
|
} // namespace mozilla
|