| /* |
| * Copyright (C) 2010 Google Inc. All rights reserved. |
| * Copyright (C) 2017-2021 Apple 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. |
| * 3. Neither the name of Apple Inc. ("Apple") nor the names of |
| * its contributors may be used to endorse or promote products derived |
| * from this software without specific prior written permission. |
| * |
| * THIS SOFTWARE IS PROVIDED BY APPLE 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 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. |
| */ |
| |
| #pragma once |
| |
| #include "AudioBufferOptions.h" |
| #include "ExceptionOr.h" |
| #include "JSValueInWrappedObject.h" |
| #include <JavaScriptCore/Forward.h> |
| #include <JavaScriptCore/GenericTypedArrayView.h> |
| #include <wtf/Lock.h> |
| #include <wtf/Vector.h> |
| |
| namespace WebCore { |
| |
| class AudioBus; |
| class WebCoreOpaqueRoot; |
| |
| class AudioBuffer : public RefCounted<AudioBuffer> { |
| public: |
| enum class LegacyPreventDetaching : bool { No, Yes }; |
| static RefPtr<AudioBuffer> create(unsigned numberOfChannels, size_t numberOfFrames, float sampleRate, LegacyPreventDetaching = LegacyPreventDetaching::No); |
| static ExceptionOr<Ref<AudioBuffer>> create(const AudioBufferOptions&); |
| // Returns nullptr if data is not a valid audio file. |
| static RefPtr<AudioBuffer> createFromAudioFileData(const void* data, size_t dataSize, bool mixToMono, float sampleRate); |
| |
| // Format |
| size_t originalLength() const { return m_originalLength; } |
| double originalDuration() const { return originalLength() / static_cast<double>(sampleRate()); } |
| float sampleRate() const { return m_sampleRate; } |
| |
| // The following function may start returning 0 if any of the underlying channel buffers gets detached. |
| size_t length() const { return hasDetachedChannelBuffer() ? 0 : m_originalLength; } |
| double duration() const { return length() / static_cast<double>(sampleRate()); } |
| |
| // Channel data access |
| unsigned numberOfChannels() const { return m_channels.size(); } |
| ExceptionOr<JSC::JSValue> getChannelData(JSDOMGlobalObject&, unsigned channelIndex); |
| ExceptionOr<void> copyFromChannel(Ref<Float32Array>&&, unsigned channelNumber, unsigned bufferOffset); |
| ExceptionOr<void> copyToChannel(Ref<Float32Array>&&, unsigned channelNumber, unsigned startInChannel); |
| |
| // Native channel data access. |
| RefPtr<Float32Array> channelData(unsigned channelIndex); |
| float* rawChannelData(unsigned channelIndex); |
| void zero(); |
| |
| // Because an AudioBuffer has a JavaScript wrapper, which will be garbage collected, it may take a while for this object to be deleted. |
| // releaseMemory() can be called when the AudioContext goes away, so we can release the memory earlier than when the garbage collection happens. |
| // Careful! Only call this when the page unloads, after the AudioContext is no longer processing. |
| void releaseMemory(); |
| |
| size_t memoryCost() const; |
| |
| template<typename Visitor> void visitChannelWrappers(Visitor&); |
| |
| bool copyTo(AudioBuffer&) const; |
| |
| enum class ShouldCopyChannelData : bool { No, Yes }; |
| Ref<AudioBuffer> clone(ShouldCopyChannelData = ShouldCopyChannelData::Yes) const; |
| |
| bool topologyMatches(const AudioBuffer&) const; |
| |
| private: |
| AudioBuffer(unsigned numberOfChannels, size_t length, float sampleRate, LegacyPreventDetaching = LegacyPreventDetaching::No); |
| explicit AudioBuffer(AudioBus&); |
| |
| void invalidate(); |
| |
| bool hasDetachedChannelBuffer() const; |
| |
| // We do not currently support having the Float32Arrays in m_channels being more than 2GB, |
| // and we have tests that we return an error promptly on trying to create such a huge AudioBuffer. |
| static constexpr uint64_t s_maxLength = (1ull << 32) / sizeof(float); |
| |
| float m_sampleRate; |
| size_t m_originalLength; |
| FixedVector<RefPtr<Float32Array>> m_channels; |
| FixedVector<JSValueInWrappedObject> m_channelWrappers; |
| bool m_isDetachable { true }; |
| mutable Lock m_channelsLock; |
| }; |
| |
| WebCoreOpaqueRoot root(AudioBuffer*); |
| |
| } // namespace WebCore |