| /* |
| * Copyright (C) 2014 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. |
| * |
| * 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. |
| */ |
| |
| #pragma once |
| |
| #if ENABLE(WEB_AUDIO) && USE(MEDIATOOLBOX) |
| |
| #include "AudioStreamDescription.h" |
| #include "CAAudioStreamDescription.h" |
| #include <JavaScriptCore/ArrayBuffer.h> |
| #include <wtf/Lock.h> |
| #include <wtf/UniqueRef.h> |
| #include <wtf/Vector.h> |
| |
| typedef struct AudioBufferList AudioBufferList; |
| |
| namespace WebCore { |
| |
| class CARingBufferStorage { |
| WTF_MAKE_FAST_ALLOCATED; |
| public: |
| virtual ~CARingBufferStorage() = default; |
| virtual bool allocate(size_t, const CAAudioStreamDescription& format, size_t frameCount) = 0; |
| virtual void deallocate() = 0; |
| virtual void* data() = 0; |
| virtual void getCurrentFrameBounds(uint64_t& startTime, uint64_t& endTime) = 0; |
| virtual void setCurrentFrameBounds(uint64_t startFrame, uint64_t endFrame) = 0; |
| virtual void updateFrameBounds() { } |
| virtual uint64_t currentStartFrame() const = 0; |
| virtual uint64_t currentEndFrame() const = 0; |
| virtual void flush() = 0; |
| virtual size_t size() const = 0; |
| }; |
| |
| class CARingBufferStorageVector final : public CARingBufferStorage { |
| public: |
| CARingBufferStorageVector(); |
| ~CARingBufferStorageVector() = default; |
| |
| private: |
| bool allocate(size_t byteCount, const CAAudioStreamDescription&, size_t) final; |
| void deallocate() final { m_buffer.clear(); } |
| void* data() final { return m_buffer.data(); } |
| void getCurrentFrameBounds(uint64_t& startTime, uint64_t& endTime) final; |
| void setCurrentFrameBounds(uint64_t startFrame, uint64_t endFrame) final; |
| uint64_t currentStartFrame() const final; |
| uint64_t currentEndFrame() const final; |
| void flush() final; |
| size_t size() const final { return m_buffer.size(); } |
| |
| struct TimeBounds { |
| TimeBounds() |
| : m_startFrame(0) |
| , m_endFrame(0) |
| , m_updateCounter(0) |
| { |
| } |
| volatile uint64_t m_startFrame; |
| volatile uint64_t m_endFrame; |
| volatile uint32_t m_updateCounter; |
| }; |
| |
| Vector<uint8_t> m_buffer; |
| Vector<TimeBounds> m_timeBoundsQueue; |
| Lock m_currentFrameBoundsLock; |
| std::atomic<int32_t> m_timeBoundsQueuePtr { 0 }; |
| }; |
| |
| class CARingBuffer { |
| WTF_MAKE_FAST_ALLOCATED; |
| public: |
| WEBCORE_EXPORT CARingBuffer(); |
| WEBCORE_EXPORT explicit CARingBuffer(UniqueRef<CARingBufferStorage>&&); |
| WEBCORE_EXPORT ~CARingBuffer(); |
| |
| WEBCORE_EXPORT static UniqueRef<CARingBuffer> adoptStorage(UniqueRef<CARingBufferStorage>&&, const CAAudioStreamDescription&, size_t frameCount); |
| |
| enum Error { |
| Ok, |
| TooMuch, // fetch start time is earlier than buffer start time and fetch end time is later than buffer end time |
| }; |
| |
| WEBCORE_EXPORT bool allocate(const CAAudioStreamDescription&, size_t frameCount); |
| WEBCORE_EXPORT void deallocate(); |
| |
| WEBCORE_EXPORT Error store(const AudioBufferList*, size_t frameCount, uint64_t startFrame); |
| |
| enum FetchMode { Copy, Mix }; |
| WEBCORE_EXPORT bool fetchIfHasEnoughData(AudioBufferList*, size_t frameCount, uint64_t startFrame, FetchMode = Copy); |
| |
| // Fills buffer with silence if there is not enough data. |
| WEBCORE_EXPORT void fetch(AudioBufferList*, size_t frameCount, uint64_t startFrame, FetchMode = Copy); |
| |
| WEBCORE_EXPORT void flush(); |
| |
| WEBCORE_EXPORT void getCurrentFrameBounds(uint64_t& startFrame, uint64_t& endFrame); |
| |
| uint32_t channelCount() const { return m_channelCount; } |
| CARingBufferStorage& storage() { return m_buffers; } |
| |
| private: |
| void updateFrameBounds(); |
| size_t frameOffset(uint64_t frameNumber) const { return (frameNumber % m_frameCount) * m_bytesPerFrame; } |
| |
| void clipTimeBounds(uint64_t& startRead, uint64_t& endRead); |
| void setCurrentFrameBounds(uint64_t startFrame, uint64_t endFrame); |
| |
| void getCurrentFrameBoundsWithoutUpdate(uint64_t& startFrame, uint64_t& endFrame); |
| void fetchInternal(AudioBufferList*, size_t frameCount, uint64_t startFrame, FetchMode); |
| |
| void initializeAfterAllocation(const CAAudioStreamDescription& format, size_t frameCount); |
| |
| uint64_t currentStartFrame() const; |
| uint64_t currentEndFrame() const; |
| |
| UniqueRef<CARingBufferStorage> m_buffers; |
| Vector<Byte*> m_pointers; |
| uint32_t m_channelCount { 0 }; |
| size_t m_bytesPerFrame { 0 }; |
| uint32_t m_frameCount { 0 }; |
| size_t m_capacityBytes { 0 }; |
| |
| CAAudioStreamDescription m_description; |
| }; |
| |
| } |
| |
| #endif // ENABLE(WEB_AUDIO) && USE(MEDIATOOLBOX) |