blob: d9abb9777a9d25eeaf242ff4fffc5ec5ef486c4e [file] [log] [blame]
/*
* Copyright (C) 2010, Google Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS BE LIABLE FOR ANY
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
* ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "config.h"
#if ENABLE(WEB_AUDIO)
#include "AudioDestinationNode.h"
#include "AudioContext.h"
#include "AudioNodeInput.h"
#include "AudioNodeOutput.h"
#include "AudioUtilities.h"
#include "DenormalDisabler.h"
#include <wtf/IsoMallocInlines.h>
namespace WebCore {
WTF_MAKE_ISO_ALLOCATED_IMPL(AudioDestinationNode);
AudioDestinationNode::AudioDestinationNode(AudioContext& context, float sampleRate)
: AudioNode(context, sampleRate)
, m_currentSampleFrame(0)
, m_isSilent(true)
, m_isEffectivelyPlayingAudio(false)
, m_muted(false)
{
setNodeType(NodeTypeDestination);
addInput(makeUnique<AudioNodeInput>(this));
}
AudioDestinationNode::~AudioDestinationNode()
{
uninitialize();
}
void AudioDestinationNode::render(AudioBus*, AudioBus* destinationBus, size_t numberOfFrames)
{
// We don't want denormals slowing down any of the audio processing
// since they can very seriously hurt performance.
// This will take care of all AudioNodes because they all process within this scope.
DenormalDisabler denormalDisabler;
context().setAudioThread(Thread::current());
if (!context().isInitialized()) {
destinationBus->zero();
setIsSilent(true);
return;
}
ASSERT(numberOfFrames);
if (!numberOfFrames) {
destinationBus->zero();
setIsSilent(true);
return;
}
// Let the context take care of any business at the start of each render quantum.
context().handlePreRenderTasks();
// This will cause the node(s) connected to us to process, which in turn will pull on their input(s),
// all the way backwards through the rendering graph.
AudioBus* renderedBus = input(0)->pull(destinationBus, numberOfFrames);
if (!renderedBus)
destinationBus->zero();
else if (renderedBus != destinationBus) {
// in-place processing was not possible - so copy
destinationBus->copyFrom(*renderedBus);
}
// Process nodes which need a little extra help because they are not connected to anything, but still need to process.
context().processAutomaticPullNodes(numberOfFrames);
// Let the context take care of any business at the end of each render quantum.
context().handlePostRenderTasks();
// Advance current sample-frame.
m_currentSampleFrame += numberOfFrames;
setIsSilent(destinationBus->isSilent());
// The reason we are handling mute after the call to setIsSilent() is because the muted state does
// not affect the audio destination node's effective playing state.
if (m_muted)
destinationBus->zero();
}
void AudioDestinationNode::isPlayingDidChange()
{
updateIsEffectivelyPlayingAudio();
}
void AudioDestinationNode::setIsSilent(bool isSilent)
{
if (m_isSilent == isSilent)
return;
m_isSilent = isSilent;
updateIsEffectivelyPlayingAudio();
}
void AudioDestinationNode::updateIsEffectivelyPlayingAudio()
{
bool isEffectivelyPlayingAudio = isPlaying() && !m_isSilent;
if (m_isEffectivelyPlayingAudio == isEffectivelyPlayingAudio)
return;
m_isEffectivelyPlayingAudio = isEffectivelyPlayingAudio;
context().isPlayingAudioDidChange();
}
} // namespace WebCore
#endif // ENABLE(WEB_AUDIO)