/* * 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 "modules/webaudio/AnalyserNode.h" #include "bindings/core/v8/ExceptionMessages.h" #include "bindings/core/v8/ExceptionState.h" #include "core/dom/ExceptionCode.h" #include "modules/webaudio/AudioNodeInput.h" #include "modules/webaudio/AudioNodeOutput.h" namespace blink { AnalyserHandler::AnalyserHandler(AudioNode& node, float sampleRate) : AudioBasicInspectorHandler(NodeTypeAnalyser, node, sampleRate, 2) { initialize(); } PassRefPtr AnalyserHandler::create(AudioNode& node, float sampleRate) { return adoptRef(new AnalyserHandler(node, sampleRate)); } AnalyserHandler::~AnalyserHandler() { uninitialize(); } void AnalyserHandler::process(size_t framesToProcess) { AudioBus* outputBus = output(0).bus(); if (!isInitialized() || !input(0).isConnected()) { outputBus->zero(); return; } AudioBus* inputBus = input(0).bus(); // Give the analyser the audio which is passing through this AudioNode. m_analyser.writeInput(inputBus, framesToProcess); // For in-place processing, our override of pullInputs() will just pass the audio data through unchanged if the channel count matches from input to output // (resulting in inputBus == outputBus). Otherwise, do an up-mix to stereo. if (inputBus != outputBus) outputBus->copyFrom(*inputBus); } void AnalyserHandler::setFftSize(unsigned size, ExceptionState& exceptionState) { if (!m_analyser.setFftSize(size)) { exceptionState.throwDOMException( IndexSizeError, (size < RealtimeAnalyser::MinFFTSize || size > RealtimeAnalyser::MaxFFTSize) ? ExceptionMessages::indexOutsideRange("FFT size", size, RealtimeAnalyser::MinFFTSize, ExceptionMessages::InclusiveBound, RealtimeAnalyser::MaxFFTSize, ExceptionMessages::InclusiveBound) : ("The value provided (" + String::number(size) + ") is not a power of two.")); } } void AnalyserHandler::setMinDecibels(double k, ExceptionState& exceptionState) { if (k < maxDecibels()) { m_analyser.setMinDecibels(k); } else { exceptionState.throwDOMException( IndexSizeError, ExceptionMessages::indexExceedsMaximumBound("minDecibels", k, maxDecibels())); } } void AnalyserHandler::setMaxDecibels(double k, ExceptionState& exceptionState) { if (k > minDecibels()) { m_analyser.setMaxDecibels(k); } else { exceptionState.throwDOMException( IndexSizeError, ExceptionMessages::indexExceedsMinimumBound("maxDecibels", k, minDecibels())); } } void AnalyserHandler::setSmoothingTimeConstant(double k, ExceptionState& exceptionState) { if (k >= 0 && k <= 1) { m_analyser.setSmoothingTimeConstant(k); } else { exceptionState.throwDOMException( IndexSizeError, ExceptionMessages::indexOutsideRange("smoothing value", k, 0.0, ExceptionMessages::InclusiveBound, 1.0, ExceptionMessages::InclusiveBound)); } } // ---------------------------------------------------------------- AnalyserNode::AnalyserNode(AbstractAudioContext& context, float sampleRate) : AudioBasicInspectorNode(context) { setHandler(AnalyserHandler::create(*this, sampleRate)); } AnalyserNode* AnalyserNode::create(AbstractAudioContext& context, float sampleRate) { return new AnalyserNode(context, sampleRate); } AnalyserHandler& AnalyserNode::analyserHandler() const { return static_cast(handler()); } unsigned AnalyserNode::fftSize() const { return analyserHandler().fftSize(); } void AnalyserNode::setFftSize(unsigned size, ExceptionState& exceptionState) { return analyserHandler().setFftSize(size, exceptionState); } unsigned AnalyserNode::frequencyBinCount() const { return analyserHandler().frequencyBinCount(); } void AnalyserNode::setMinDecibels(double min, ExceptionState& exceptionState) { analyserHandler().setMinDecibels(min, exceptionState); } double AnalyserNode::minDecibels() const { return analyserHandler().minDecibels(); } void AnalyserNode::setMaxDecibels(double max, ExceptionState& exceptionState) { analyserHandler().setMaxDecibels(max, exceptionState); } double AnalyserNode::maxDecibels() const { return analyserHandler().maxDecibels(); } void AnalyserNode::setSmoothingTimeConstant(double smoothingTime, ExceptionState& exceptionState) { analyserHandler().setSmoothingTimeConstant(smoothingTime, exceptionState); } double AnalyserNode::smoothingTimeConstant() const { return analyserHandler().smoothingTimeConstant(); } void AnalyserNode::getFloatFrequencyData(DOMFloat32Array* array) { analyserHandler().getFloatFrequencyData(array, context()->currentTime()); } void AnalyserNode::getByteFrequencyData(DOMUint8Array* array) { analyserHandler().getByteFrequencyData(array, context()->currentTime()); } void AnalyserNode::getFloatTimeDomainData(DOMFloat32Array* array) { analyserHandler().getFloatTimeDomainData(array); } void AnalyserNode::getByteTimeDomainData(DOMUint8Array* array) { analyserHandler().getByteTimeDomainData(array); } } // namespace blink